Python编码爬坑指南(必看)


Posted in Python onJune 10, 2016

自己最近有在学习python,这实在是一门非常短小精悍的语言,很喜欢这种语言精悍背后又有强大函数库支撑的语言。可是刚接触不久就遇到了让人头疼的关于编码的问题,在网上查了很多资料现在在这里做一番总结,权当一个记录也为后来的兄弟姐妹们服务,如果可以让您少走一些弯路本人将倍感荣幸。

先来描述下现象吧:

import os
for i in os.listdir("E:\Torchlight II"):
  print i

代码很简单我们使用os的listdir函数遍历了E:\Torchlight II这个目录(Torchlight ?! :)),由于这个目录下有些文件是以中文命名的,所以在最后print结果时出现了乱码,像这样:

Python编码爬坑指南(必看)

那么问题出在哪儿呢? 别急,我们一点一点来分析它。

这里这里我们几乎能够肯定的知道问题是出在:

This means that the python console app can't write the given character to the console's encoding.
More specifically, the python console app created a _io.TextIOWrapperd instance with an encoding that cannot represent the given character.
sys.stdout --> _io.TextIOWrapperd --> (your console)

看到这里不知你是否与我想的一样,能不能去设置console的编码,将其设置为能够理解中文字符的编码不就可以正常的显示出中文了吗?等等,让我们在多Google一会儿,

Python determines the encoding of stdout and stderr based on the value of the LC_CTYPE variable, but only if the stdout is a tty. So if I just output to the terminal, LC_CTYPE (or LC_ALL) define the encoding. However, when the output is piped to a file or to a different process, the encoding is not defined, and defaults to 7-bit ASCII.

更详细的说明如下:

1). When Python finds its output attached to a terminal, it sets the sys.stdout.encoding attribute to the terminal's encoding. The print statement's handler will automatically encode unicode arguments into str output.
2). When Python does not detect the desired character set of the output, it sets sys.stdout.encoding to None, and print will invoke the "ascii" codec.

嚯嚯,看来刚才的想法是可行的只是不太优雅罢了,因为我们得去修改系统的设置。事实上上面的论述是基于linux环境的,在linux下可能需要我们去更改某个环境变量的值(LC_CTYPE or LANG);如果我们是在windows下面的话,console的编码设置是跟操作系统的区域设置相关的。比如在中文的win7环境下,console默认的编码就是GBK(cp936)。你可以试试下面的代码:

import locale
print locale.getdefaultlocale()[1]

console的编码不好设置了那能否对stdout.out.encoding进行设置以达到我们的目的呢?很遗憾,答案是否定的,这家伙压根就是只读的:

Python编码爬坑指南(必看)

没有办法了么?不会,其实我们离成功已经很近了,来,根据上面检索到的那些资料分析整理下看看我们现在掌握到的情况都有哪些:

 

1). console不能正常显示中文,console的编码是由操作系统决定的(windows环境下);
 2). 我的操作系统是win7中文版(GBK),enc = locale.getdefaultlocale()[1];
 3). console的编码决定了sys.stdout.encoding的取值,sys.stdout.encoding = utf-8;
 4). 从操作系统枚举目录(E:\Torchlight II)列表返回的字符串也是GBK编码

 是不是已经看出问题来了。最上面截图中那么奇奇怪怪的问号尖角符号就是因为字符串本身是按照gbk进行编码的,但是由于sys.stdout.encoding = utf-8,导致print会按照utf-8对input的数据进行encode从而转换为unicode字符。这,当然错误了。原因已经清楚了,来改改代码吧:

import os
for i in os.listdir("E:\Torchlight II"):
  print i.decode('gbk')

在代码中我们手动告诉了python对读入的字符串按章gbk编码来进行解码,而这一个动作之后数据已经是标准的unicode字符了,可以放心的交给print去打印输出了(即使这会儿sys.stdout.encoding = utf-8):

Python编码爬坑指南(必看)

 ps:

实际在google中还查到过很多相关的类似编码的问题,比如这里的,还有这里的。虽然问题的样子千变万化并且解决方式多种多样甚至是python自己的特定解决方式,比如这里。但这些问题本质都是一样的都是关于字符的编码和解码,搞清楚了其中的本质所有问题都能够迎刃而解。

以上这篇Python编码爬坑指南(必看)就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python实现Linux下守护进程的编写方法
Aug 22 Python
Python模拟登录验证码(代码简单)
Feb 06 Python
使用Python从零开始撸一个区块链
Mar 14 Python
详解Ubuntu16.04安装Python3.7及其pip3并切换为默认版本
Feb 25 Python
Python 中的参数传递、返回值、浅拷贝、深拷贝
Jun 25 Python
Python解决pip install时出现的Could not fetch URL问题
Aug 01 Python
pytorch实现用CNN和LSTM对文本进行分类方式
Jan 08 Python
pytorch 实现删除tensor中的指定行列
Jan 13 Python
PIL.Image.open和cv2.imread的比较与相互转换的方法
Jun 03 Python
tensorflow下的图片标准化函数per_image_standardization用法
Jun 30 Python
python 实现定时任务的四种方式
Apr 01 Python
Python实现信息轰炸工具(再也不怕说不过别人了)
Jun 11 Python
浅析Python中的for 循环
Jun 09 #Python
Python多层嵌套list的递归处理方法(推荐)
Jun 08 #Python
Python-嵌套列表list的全面解析
Jun 08 #Python
PYTHON压平嵌套列表的简单实现
Jun 08 #Python
Python用Bottle轻量级框架进行Web开发
Jun 08 #Python
浅谈Python数据类型之间的转换
Jun 08 #Python
浅谈python 四种数值类型(int,long,float,complex)
Jun 08 #Python
You might like
php 设计模式之 单例模式
2008/12/19 PHP
PHP开发需要注意的安全问题
2010/09/01 PHP
php程序的国际化实现方法(利用gettext)
2011/08/14 PHP
使用配置类定义Codeigniter全局变量
2014/06/12 PHP
thinkphp表单上传文件并将文件路径保存到数据库中
2016/07/28 PHP
PHP实现求连续子数组最大和问题2种解决方法
2017/12/26 PHP
Laravel中unique和exists验证规则的优化详解
2018/01/28 PHP
PHP封装的非对称加密RSA算法示例
2018/05/28 PHP
thinkPHP5框架中widget的功能与用法详解
2018/06/11 PHP
获取dom元素那些讨厌的位置封装代码
2010/06/23 Javascript
js 获取和设置css3 属性值的实现方法
2013/05/06 Javascript
Jquery同辈元素选中/未选中效果的实例代码
2013/08/01 Javascript
jquery为页面增加快捷键示例
2014/01/31 Javascript
javascript中通过arguments参数伪装方法重载
2014/10/08 Javascript
jQuery中width()方法用法实例
2014/12/24 Javascript
html的DOM中Event对象onblur事件用法实例
2015/01/21 Javascript
jquery.fastLiveFilter.js实现输入自动过滤的方法
2015/08/11 Javascript
jQuery抛物线运动实现方法(附完整demo源码下载)
2016/01/08 Javascript
Bootstrap多级导航栏(级联导航)的实现代码
2016/03/08 Javascript
jQuery操作属性和样式详解
2016/04/13 Javascript
node.js 中国天气预报 简单实现
2016/06/06 Javascript
jQuery ui autocomplete选择列表被Bootstrap模态窗遮挡的完美解决方法
2016/09/23 Javascript
使用snowfall.jquery.js实现爱心满屏飞的效果
2017/01/05 Javascript
Node+Express+MongoDB实现登录注册功能实例
2017/04/23 Javascript
详解Howler.js Web音频播放终极解决方案
2020/08/23 Javascript
使用webpack5从0到1搭建一个react项目的实现步骤
2020/12/16 Javascript
[50:28]LGD女子学院第三期 DOTA2复仇之魂教学
2013/12/24 DOTA
python分割和拼接字符串
2013/11/01 Python
Python函数可变参数定义及其参数传递方式实例详解
2015/05/25 Python
ML神器:sklearn的快速使用及入门
2019/07/11 Python
Django 迁移、操作数据库的方法
2019/08/02 Python
音乐系毕业生自荐信
2013/10/27 职场文书
音乐幼师求职信
2014/07/09 职场文书
公司安全管理制度范本
2015/08/05 职场文书
《鲁滨逊漂流记》之六读后感(4篇)
2019/09/29 职场文书
vue3语法糖内的defineProps及defineEmits
2022/04/14 Vue.js