一文透彻掌握 Python 编码问题
一、当我说字符时,我在说什么?
当我们提起字符串时,每个程序员都能理解到,我们说的是一个字符序列。但是,当我们说字符时,很多人就困惑了。
写在纸上的字符很容易辨识,但是为了将不同的字符在计算机中标识出来,人类发明了unicode字符。简单讲,unicode可以看成是一个标准的函数,它将一个具体的字符映射成0-1114111之间的一个数字,这个数字叫做码位。通常,码位用十六进制表示,并且前面会加上“U+”的字样。例如,字母A的码位是U+0041。
按道理说,我们在计算机中,用unicode的码位来代表字符就很完美了。实际上,python3中的str对象和python2中的unicode对象在内存中就是用码位来表示字符的。
但是,由于全世界的字符比较多,导致表示码位的数字也要用long或者int这样的数据类型表示,每个字符都要占固定的几个字节。在存储到磁盘或者通过网络进行传输时,比较浪费空间。于是,聪明的人类又搞了一个函数,这个函数将一个码位映射成字节序列。映射的目的是减少占用的空间。这个函数就是编码。也就是说,编码是在码位和字节序列之间转换时使用的算法。
比如大写字母A(U+0041),使用UTF-8编码后是\x41,这里\x表示一个字节,字节的值是41。我们看到,如果用U+0041 这个整数代表大写字母A,需要4个字节,因为一个整数就是用4个字节表示的,经过编码后,只占用了一个字节,达到了减少减少空间的目的。
这里提示一下,下文中,当我们再说到码位时,可以将其简单想象成一个int。
二、python3中码位和编码是如何表示的
在python3的代码中,str类型的对象就是用码位表示的字符串, 编码后的字节序列可以用bytes类型的对象表示。如下所示:

可以将bytes类型的对象看成一个数组,切片啥的都不在话下,里面的元素是介于0-255(含)之间的整数。从python2.6起,新增一个类似的类型,bytearray。它和bytes很像,不同之处有以下两点:
1、没有字面量句法,看图:

上图是bytes对象的字面量创建方法。bytearray没有类似的构造方法,它只能这样获得:

2、bytes不可变,bytearray可变
当我们print一个bytes对象时,常常会看到这种情况:b'caf\xc3\t'看起来有点乱,让我们来观察一下:
b表示你print的是bytes对象
caf表示三个字节,这几个字节中的值就是caf三个字符的ascii码值,这里直接用caf三个字符表示了。
\xc3表示这个字节中的值是十六进制的c3,无法用ascii码值表示,所以这里用了两个字节的十六进制数表示。\t表示,这个字节的值是tab字符,这里就用转义字符来表示了。
三、python中的编解码器
python有100多种编解码器!第一次知道这个消息,我很震惊,人类真是喜欢折腾啊。下面,让我们一起来欣赏一下几个常用的编解码器对一些字符的编码:

四、处理常见的编解码错误
在用python进行编解码时,经常发生各种错误。很多人的办法就是各种google各种试,搞定之后就不再管了。我自己之前就是这样。但更系统的办法就是理解常见的错误类型,在遇到时可以按步骤地去解决问题。下面我们就来看看常见的三类错误。
UnicodeEncoderError
当你用了某个编码器将unicode字符进行编码输出时,如果这个编码器中没有包含某些要编码的unicode字符,就会发生UnicodeEncoderError。这种情况下,通常需要去改变所用的编码器。简单讲就是在将unicode进行encode时发生了error
UnicodeDecodeError
在将一个字节序列用指定的解码器解码成unicode时,如果这个字节序列不符合解码器的要求,就会发生UnicodeDecoderError。这里的不符合要求有两种情况,一种是字节序列错误的,一种就是用的解码器不合适。
SyntaxError
python3默认使用UTF-8编码源码,python2则默认使用ASCII。如果加载的.py文件中包含UTF-8之外的数据,而且没有声明编码,就会发生SyntaxError。
处理编解码的最佳实践是,明确指定encoding字段,显式声明所用的编解码器。
五、几种编码默认值的区别
locale.getpreferredencoding()
这个设置是打开文本文件时,默认使用的解码器。如果open()文件时没有指定解码器,并且发生了错误,就要检查一下这个值。如下是在我的电脑上测试的结果

赶紧看看自己的电脑是什么编码吧。
sys.getdefaultencoding()
当在python程序内,在字节序列和字符串之间转换时,默认使用这个编码。python默认的是UTF-8。
sys.getfilesystemencoding()
这个是文件名默认的编解码器,注意:不是文件内容,只是文件名称。open()里面传入文件名给python,这时的文件名是unicode字符串,python是用这个编码器对名字进行编码,转成字节序列后再去文件系统中查找的。如下所示,是我电脑上的结果:
注意和上面的locale.getpreferredencoding()进行一下对比哈。
sys.stdout.encoding和sys.stdin.coding
这时python标准输入输出使用的默认编码,在我的电脑上是这样的:

我们经常发现中文输出乱码时,原因要从两头找,一头就是python默认输出时使用的编码器,一头就是显示的控制台使用的解码器,理论上,只要二者一致,就不会发生错误。
一文透彻掌握 Python 编码问题的更多相关文章
- Python之路3【知识点】白话Python编码和文件操作
Python文件头部模板 先说个小知识点:如何在创建文件的时候自动添加文件的头部信息! 通过:file--settings 每次都通过file--setings打开设置页面太麻烦了!可以通过:View ...
- python编码规范
python编码规范 文件及目录规范 文件保存为 utf-8 格式. 程序首行必须为编码声明:# -*- coding:utf-8 -*- 文件名全部小写. 代码风格 空格 设置用空格符替换TAB符. ...
- 【转】python编码规范
http://blog.csdn.net/willhuo/article/details/49300441 决定开始Python之路了,利用业余时间,争取更深入学习Python.编程语言不是艺术,而是 ...
- python 编码规范整理
PEP8 Python 编码规范 一 代码编排1 缩进.4个空格的缩进(编辑器都可以完成此功能),不要使用Tap,更不能混合使用Tap和空格.2 每行最大长度79,换行可以使用反斜杠,最好使用圆括号. ...
- python编码详解--转自(Alex的博客)
原文地址:http://www.cnblogs.com/alex3714/articles/7550940.html 编码回顾 在备编码相关的课件时,在知乎上看到一段关于Python编码的回答 这哥们 ...
- python书籍推荐:python编码推荐(高清完整pdf)
目录INF-qa Python 编码规范................................................................................ ...
- PEP8 - Python编码规范
PEP8 - Python编码规范 PEP8 规范 官方文档:https://www.python.org/dev/peps/pep-0008/ PEP8 规范 中文翻译:http://www.cnb ...
- python经典书籍推荐:python编码规范
目录INF-qa Python 编码规范................................................................................ ...
- 说说Python编码规范
前言 已有近两个月没有发表过文章了,前段时间外甥和女儿过来这边渡暑假,平常晚上和周末时间都陪着她们了,趁这个周末有空,再抽空再把这块拾起来. 这么久没写了,再次拿起键盘,想想,发表些什 ...
随机推荐
- Oracle数据库弱口令解密
1.首先我们先看下Oracle加密的格式. 接下来我们把他头和尾部删除,中间加: 通过py脚本来进行完成 转换完成后输出成这种格式 之后直接丢进kali里面,用john --w=字典文件 + 转换文件 ...
- nginx 增加认证
1.检查工具是否安装,如果未安装则使用yum安装 #htpasswd 有以上输出表示已经安装,如果没有按装,使用如下命令安装: #yum -y install httpd-tools 2.htpass ...
- Anaconda Spyder 常用快捷键
Ctrl+1 注释.取消注释 Ctrl+4/5 块注释 / 取消块注释 Ctrl+D 删除一行 Ctrl+L 转到行 Ctrl+G/左键 查找函数定义 F9 运行选中代码 F12 断点 / 取消断点 ...
- [转帖]盖茨辉煌后将归隐 DOS之父仍为生计打拼(图)
盖茨辉煌后将归隐 DOS之父仍为生计打拼(图) http://www.sina.com.cn 2007年12月10日 09:43 新浪科技 https://tech.sina.com.cn/it/2 ...
- java如何防止反编译(转)
出处: java如何防止反编译 一些防止java代码被反编译的方法 综述(写在前面的废话) Java从诞生以来,其基因就是开放精神,也正因此,其可以得到广泛爱好者的支持和奉献,最终很快发展壮大,以至于 ...
- ASP.NET Core MVC里面Razor如何获取URL参数
原文:ASP.NET Core MVC里面Razor如何获取URL参数 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https:// ...
- js判断一个 object 对象是否为空
方法一:使用for...in for...in... 遍历属性,为真则为“非空数组”:否则为“空数组” for (var i in obj) { return true // 如果不为空,则会执行到 ...
- Unknown column '字段名' in 'field list' 错误解决方案
当后台报这个错误的时候,说明实体类中的字段名出错了,解决方案: 1,检查数据库中的字段名与实体类中的字段名是否一致,特别要注意单词字母, 2,检查数据库中的字段是否与实体类中的字段一致.比如数据库中没 ...
- python之项目依赖管理
生成所有依赖清单 requirements.txt 1. pipreqs 工具 安装) pip install pipreqs 执行生成依赖列表命令) pipreqs ./ 完善版本: pipreq ...
- ES6入门五:箭头函数、函数与ES6新语法
箭头函数的基本用法与特点 函数与ES6新语法 一.箭头函数的基本用法与特点 声明箭头函数采用声明变量和常量的关键字:var.let.const 箭头函数的参数:没有参数用"()"空 ...