python cmd 窗口 中文乱码 解决方法 (附:打印不同颜色)
前言
在 python 开发中,有时候想通过cmd窗口来和用户交互,比如显示信息之类的,会比自己创建 GUI 来的方便,但是随之而来的就是编码乱码问题
下面例子在 python2 和 python3 中都可以运行,也可以在其它 .py 中通过 import os;os.startfile(ur"xxx.bat") 来运行
之前一直遇到一个问题,通过双击 bat 文件来运行,可以不用转码,只要 cmd 窗口的活动页编码是 936 就可以了,但是在其它 py 文件中运行那个 bat,就会出现乱码,所以转码后可以一保正常
双击 .bat 来运行 py
@echo off
chcp 936>nul
python d:\test.py exit
# -*- coding: utf-8-*- import sys import os input_ = input zh_coding = 'cp936' utf_8 = 'utf-8' # before other codes if sys.version_info.major < 3: reload(sys) sys.setdefaultencoding(utf_8) input_ = raw_input # begin your code def to936(utf8): if sys.version_info.major < 3: utf8 = utf8.decode().encode(zh_coding) return utf8 def toUtf8(zh936): if sys.version_info.major < 3: zh936 = zh936.decode(zh_coding).encode(utf_8) return zh936 os.system('@echo off') # 转换cmd窗口的编码集,不管cmd窗口的默认值是什么都显示正确 os.system('chcp 936 >nul') print(to936(u'中文简体(encoding)')) print(to936(u'中文繁體(encoding)')) # 在 cmd 窗口 的默认值的默认代码页不是 936 的情况下会报错。 # 如果是 936,通过双击 .bat 运行是正常的, # 如果在其它 py 中通过 os.startfile(), subprocess, os.system() # 等模块来运行那个bat,会显示乱码 try: print(u'中文简体(not encoding)') print(u'中文繁體(not encoding)') except Exception as e: os.system(to936(u'echo 解码错误')) print(e) i = input_(to936(u'输入你的中文名字:')) i = toUtf8(i) msg = to936(u'你的中文名字是:%s' % i) print(msg) os.system('pause && exit')
打印不同颜色
win 的 cmd 终端也可以像linux的终端一样,打印不同行颜色,同一行不同颜色
这里有个个方法:https://www.cnblogs.com/linyfeng/p/8286506.html#4304331
不过,在我使用 waf 的过程中,发现 waf 在win cmd 窗口打印颜色更全面跟通用的模块,使用发法和linux的一致
waf 是开源的,我从里面提取了相关的模块,可以集成到自己 lib 中,waf 教程和下载:https://waf.io/book/
而且,使用 waf 的模块,在 python 2 中只是print 的时候,不用转码,但是input的时候还是要转回utf-8
waf 从 git 上下载的 waflib 兼容 python 2 和 3
waflib模块中提取的文件(注意,这些文件的编写,语法缩进是 tab,而不是四个空格,所以最好自己把 tab 替换成四个空格,不然通过编译出来的 pyc 可能不能用):
__init__.py
ansiterm.py
Errors.py
Utils.py
# -*- coding: utf-8-*- import sys import os input_ = input zh_coding = 'cp936' utf_8 = 'utf-8' # before other codes if sys.version_info.major < 3: reload(sys) sys.setdefaultencoding(utf_8) # begin your code ################################################### from waflib import ansiterm if not os.environ.get('NOSYNC',False): if sys.stdout.isatty()and id(sys.stdout)==id(sys.__stdout__): sys.stdout=ansiterm.AnsiTerm(sys.stdout) if sys.stderr.isatty()and id(sys.stderr)==id(sys.__stderr__): sys.stderr=ansiterm.AnsiTerm(sys.stderr) os.system('@echo off') # 转换cmd窗口的编码集,不管cmd窗口的默认值是什么都显示正确 os.system('chcp 936 >nul') # 颜色 print(u'\033[32m绿色\033[31m红色\033[0m') print(u'中文简体(not encoding)') print(u'中文繁體(not encoding)') os.system('pause && exit')
上面没有 input 例子,如果要用,参考上上个例子
主要添加了以下代码,之后 print 就像通用的一样使用就可以了
from waflib import ansiterm if not os.environ.get('NOSYNC',False): if sys.stdout.isatty()and id(sys.stdout)==id(sys.__stdout__): sys.stdout=ansiterm.AnsiTerm(sys.stdout) if sys.stderr.isatty()and id(sys.stderr)==id(sys.__stderr__): sys.stderr=ansiterm.AnsiTerm(sys.stderr)
print 颜色 格式参考:https://www.cnblogs.com/pupilheart/p/9704943.html