Python编解码问题及文本文件处理方法详解

作者:自动化代码美学 时间:2021-04-13 07:52:06 

编解码器

在字符与字节之间的转换过程称为编解码,Python自带了超过100种编解码器,比如:

  • ascii(英文体系)

  • gb2312(中文体系)

  • utf-8(全球通用)

  • latin1

  • utf-16

编解码器一般有多个别名,比如utf8、utf-8、U8。

这些编解码器可以传给open()、str.encode()、bytes.decode()等函数的encoding参数。

UnicodeEncodeError

多数非UTF编解码器(比如cp437)只能处理Unicode字符的一小部分子集。把字符转换成字节时,如果目标编码中没有定义这个字符,那么就会抛出UnicodeEncodeError异常。

处理方式一:使用utf8编码。

处理方式二:添加errors参数:


# 忽略 如b'So Paulo'
city.encode("cp437", errors="ignore")
# 替换为? 如b'S?o Paulo'
city.encode("cp437", errors="replace")
# 替换为XML实体 如b'São Paulo'
city.encode("cp437", errors="xmlcharrefreplace")

UnicodeDecodeError

把字节转换为字符时,遇到无法转换的字节时会抛出UnicodeDecodeError异常。这是因为不是每个字节都包含有效的ASCII字符,也不是每个字符都是有效的UTF-8。

处理方式也有两种,跟上面一样。

SyntaxError

Python3默认使用UTF-8编码源码。如果加载的.py模块中包含UTF-8之外的数据,而且没有声明编码,就会抛出SyntaxError异常。

处理方式是在文件顶部添加coding注释:


# coding: cp1252

但是这个办法并不好,最好还是找到这些报错字符,把它们转换为UTF-8。

从网上直接复制代码到IDE中执行经常会报这个错。

处理文本文件

Unicode三明治:

Python编解码问题及文本文件处理方法详解

在程序中尽量少接触二进制,把字节解码为字符,只处理字符串对象。比如在Django中,view应该输出Unicode字符串,Django会负责把响应数据编码成字节序列,而且默认使用UTF-8编码。

Python内置的open函数就是采用了这个原则,在读取文件时会做必要的解码,以文本模式写入文件时会做必要的编码。

文件乱码

Windows更容易遇到这个问题,因为Windows并不是统一的UTF-8编码,比如在Windows10中:


>>> open("cafe.txt", "w", encoding="utf8").write("café")
4
>>> open("cafe.txt").read()
'caf茅'

写入文件时指定了utf8,但是读取文件没有指定,Python就会使用系统默认编码:


>>> import locale
# 打开文件用这个
# 如果没有设置PYTHONENCODING环境变量,sys.stdout/stdin/stderr也用这个
>>> locale.getpreferredencoding()
'cp936'

cp936把最后一个字节解码成了茅而不是é。


>>> import sys
# 二进制数据和字符串之间转换用这个
>>> sys.getdefaultencoding()
'utf-8'

>>> import sys
# 文件名(不是文件内容)用这个
>>> sys.getfilesystemencoding()
'utf-8'

GNU/Linux或Mac OS X不会遇到这个问题,因为多年来它们的默认编码都是UTF-8。

解决办法是一定不能依赖系统默认编码,打开文件时始终应该明确传入encoding=参数,因为不同的设备使用的默认编码可能不同,有时隔一天也会发生变化。

小结

本文介绍了Python的编解码器,以及可能出现的UnicodeEncodeError、UnicodeDecodeError、SyntaxError问题,然后给出了Python的open函数处理文本文件的原则,最后对Windows容易出现的文件乱码问题进行了说明。

参考资料:

《流畅的Python》

来源:https://www.cnblogs.com/df888/p/14901346.html

标签:python,编码,解码
0
投稿

猜你喜欢

  • JavaScript setTimeout和setInterval的使用方法 说明

    2023-08-31 10:48:19
  • ASP中生成文件的两种方式

    2007-08-30 09:49:00
  • js实现GIF动图分解成多帧图片上传

    2023-08-15 00:06:38
  • CSS缩写可以有效的缩小CSS文件的体积

    2009-01-13 18:06:00
  • python GUI库图形界面开发之PyQt5拖放控件实例详解

    2023-04-26 08:43:24
  • 解读HTML:大厦的基石

    2008-12-01 12:57:00
  • oracle下实现恢复一个丢失的数据文件的代码

    2009-03-02 11:02:00
  • python next()和iter()函数原理解析

    2023-04-05 21:58:50
  • Python argparse模块应用实例解析

    2023-08-29 12:44:52
  • 详解Django中的ifequal和ifnotequal标签使用

    2023-06-24 05:07:04
  • spyder快捷键与python符号化输出方式

    2023-08-23 11:58:24
  • Python参数传递机制传值和传引用原理详解

    2023-07-28 11:16:02
  • css设计小巧三条

    2008-01-21 13:04:00
  • 说说CSS的优先权 考虑CSS的继承与层叠

    2008-12-11 13:33:00
  • SQL Server查询速度慢原因及优化方法

    2008-12-03 15:19:00
  • PHP addslashes()函数讲解

    2023-06-04 04:28:24
  • opencv实现图像旋转效果

    2023-07-17 13:28:40
  • Python如何快速上手? 快速掌握一门新语言的方法

    2023-05-07 12:00:48
  • 动态提示的select下拉框

    2007-12-02 14:54:00
  • asp如何读取Access数据库的表名?

    2009-11-18 20:48:00
  • asp之家 网络编程 m.aspxhome.com