奇怪的编码错误,当被称为可执行vs shell脚本时的不同行为

时间:2015-05-16 05:11:00

标签: python shell utf-8

我有一个脚本,告诉我的笔记本电脑剩余多少电池。我想要一个通用的解决方案python 2& python 3,如果可能的话。通过我自己的修修补补,我似乎只能使用python 2或python 3来工作。这是代码:

#battery.py
#!/usr/bin/env python
# coding=UTF-8

import math
import subprocess

p = subprocess.Popen(["ioreg", "-rc", "AppleSmartBattery"], stdout=subprocess.PIPE)
output = p.communicate()[0]

o_max = [l for l in output.splitlines() if b'MaxCapacity' in l][0]
o_cur = [l for l in output.splitlines() if b'CurrentCapacity' in l][0]

b_max = float(o_max.rpartition(b'=')[-1].strip())
b_cur = float(o_cur.rpartition(b'=')[-1].strip())

charge = b_cur / b_max
charge_threshold = int(math.ceil(10 * charge))

# Output

total_slots, slots = 10, []
filled = int(math.ceil(charge_threshold * (total_slots / 10.0))) * u"\u25CF"
empty = (total_slots - len(filled)) * u"\u25CB"

#out = (filled + empty).decode('utf-8')
out = u' '.join((filled, empty)).encode('utf8').strip()
import sys

color_green = '%{[32m%}'
color_yellow = '%{[1;33m%}'
color_red = '%{[31m%}'
color_reset = '%{[00m%}'
color_out = (
    color_green if len(filled) > 6
    else color_yellow if len(filled) > 3
    else color_red
)

# out = color_out + out + color_reset #works with py2
out = color_out + out.decode('utf-8','ignore') + color_reset #works with py3
sys.stdout.write(out)

当通过python exectubales pythonpython3运行时,脚本会按预期输出:

$ python batcharge.py
%{%}●●●●●●●●●●%{%}%

$ python3 batcharge.py
%{%}●●●●●●●●●●%{%}%

但是当shell调用文件在我的提示符中显示其输出时,我从python2得到一个回溯错误:

Traceback (most recent call last):
  File "/usr/local/bin/batcharge.py", line 40, in <module>
    sys.stdout.write(out)
UnicodeEncodeError: 'ascii' codec can't encode characters in position 9-18: ordinal not in range(128)

当我使用带有python3的virtualenv作为默认python时,它工作正常。请帮忙。

2 个答案:

答案 0 :(得分:2)

我使用一个使用try / except子句的大杂烩解决方案来使用python2和python3。在这里我尝试python 2代码,如果它抛出异常,我使用python3代码:

[ 1 ]

它非常不优雅,最终不是pythonic,IMO。我愿意接受更优雅的解决方案。

答案 1 :(得分:0)

而不是:

sys.stdout.write(out)

使用此:

sys.stdout.buffer.write(out)