全國(guó)咨詢(xún)/投訴熱線:400-618-4000

首頁(yè)技術(shù)文章正文

5分鐘戰(zhàn)勝人工智能+Python字符編碼

更新時(shí)間:2017-11-10 來(lái)源:黑馬程序員 瀏覽量:

1. 本文不談復(fù)雜的理論,就經(jīng)驗(yàn)教你字符處理八字真言:確定編碼,同類(lèi)交互。

2. 文章針對(duì)Python 2.7,主要因?yàn)?對(duì)的編碼已經(jīng)有了很大的改善并且實(shí)際原理一樣,更改一下操作命令即可。

3. 了解完本文,你可以輕松解決文字處理,特殊平臺(tái)(Windows?)下的編碼,爬蟲(chóng)編碼等問(wèn)題。

閱讀建議

本文分為如下幾個(gè)部分:

? 原理

? 具體操作

? 建議的使用習(xí)慣

? 疑難問(wèn)題解答

如果想要了解我給出的使用習(xí)慣,可以直接跳到建議的使用習(xí)慣。

如果只想要解決相關(guān)問(wèn)題可以直接跳到疑難問(wèn)題解答。

希望本文能夠幫到你。

一. 原理

為了理解方便,這里不談理論只做類(lèi)比,具體想要進(jìn)一步了解各種編碼的理論的百度一下好了。

1. 首先說(shuō)一下我們?yōu)槭裁磿?huì)碰到各式各樣的編碼問(wèn)題:

? 因?yàn)槲覀儧](méi)有統(tǒng)一編碼

? 因?yàn)槲覀儧](méi)有用對(duì)命令(傳對(duì)數(shù)據(jù))

2. 再說(shuō)一下編碼是什么,Python的編碼看似復(fù)雜,實(shí)際上可以看做只有兩類(lèi)編碼:Unicode,二進(jìn)制

? Unicode 相信都很熟悉:,就是\u0000這樣的

? 二進(jìn)制編碼也很簡(jiǎn)單,就是\x00\x00這樣的,平常看到的utf-8,cp936都是二進(jìn)制編碼

? 二進(jìn)制編碼是具象的,10001100原樣就可以存儲(chǔ),而Unicode是抽象的,不能這樣存

#coding=utf8

# Unicode編碼演示

print('Unicode:')

print(repr('Unicode編碼'))`

# 二進(jìn)制編碼演示

print(u'二進(jìn)制編碼:')

print(repr('Unicode編碼'))`

# 這里只是看個(gè)樣子,代碼不必去深究

3. 再說(shuō)怎么做,就是只有同種編碼之間才可以操作

? 舉個(gè)簡(jiǎn)單的類(lèi)比

就把一串?dāng)?shù)據(jù)比為烤鴨,我們作為人和鴨子不同種看待烤鴨的態(tài)度完全不一樣。

我們看到的是晚上的配菜,鴨子看到的是自己二舅。

如果我用鴨子的眼光逛烤鴨店的時(shí)候就會(huì)出問(wèn)題。

因?yàn)槲以诳绝喌昕吹搅硕际亲约旱亩恕?/p>

? 這里說(shuō)的同種就是我們熟悉的各種編碼方式:utf-8,unicode,ucs-bom

? 這也就是編碼問(wèn)題的核心,非常重要。

4. 最后說(shuō)一下Python的環(huán)境

? 本身代碼是用Ascii解碼的,文件里有Ascii無(wú)法解碼的內(nèi)容的話(huà)要告知Python怎么解碼

? 內(nèi)部大量命令都是默認(rèn)接受Unicode

# 告知的命令就是下面這一行,刪掉就會(huì)報(bào)錯(cuò)

#coding=utf8

print(u'測(cè)試編碼')

二. 具體操作

拿到各種編碼的內(nèi)容自然是不用說(shuō),那么如果我們想要自己構(gòu)造怎么做呢,看下面:

#coding=utf8

# 字符串前面加u會(huì)默認(rèn)構(gòu)造出Unicode的字符串

unicodeString = u'Unicode字符串'

# 字符串前面什么都不加會(huì)構(gòu)造出默認(rèn)編碼(首行限定了現(xiàn)在的utf8)的字符串

utf8String = 'Utf-8字符串'

# 當(dāng)然,沒(méi)有首行,默認(rèn)的編碼是Ascii

那么他們之間怎么轉(zhuǎn)換呢,同樣很簡(jiǎn)單:

# 接上一段程序

# Unicode轉(zhuǎn)化為二進(jìn)制編碼中的一種:utf8

unicodeString.encode('utf8')

# 二進(jìn)制編碼根據(jù)自己的編碼種類(lèi)轉(zhuǎn)化為Unicode

utf8String.decode('utf8')

# 如果二進(jìn)制編碼中混進(jìn)了奇怪的東西可以根據(jù)需求用特殊的decode策略

print(repr('u8字\x00符串'.decode('utf8', 'replace')))

那么怎么樣會(huì)出現(xiàn)問(wèn)題呢:

# 接著上一段程序說(shuō)

# 如果我們把他們轉(zhuǎn)化成同樣的編碼方式就可以操作(例如相加)

print(repr(unicodeString + utf8String.decode('utf8')))

print(repr(unicodeString.encode('utf8') + utf8String))

# 但如果不轉(zhuǎn)化,當(dāng)然就會(huì)出現(xiàn)滿(mǎn)世界的烤鴨二舅啦

unicodeString + utf8String

# 所以另一方面也發(fā)現(xiàn),編碼轉(zhuǎn)換是需要我們告訴程序怎么做的

# 所有`decode`操作都會(huì)生成Unicode編碼,這是為了方便我之前說(shuō)的大量接受Unicode的內(nèi)部命令

所以我們需要確定程序使用的編碼,這是我們需要告訴程序的東西

? 一方面在操作字符串的時(shí)候確定是同種編碼

? 另一方面在使用非自己寫(xiě)的命令時(shí),一般使用Unicode,或者使用接收二進(jìn)制編碼的命令

#coding=utf8

# 這里拿寫(xiě)入文件舉例

# 一般使用Unicode

with open('Unicode.txt', 'w') as f: f.write(u'Unicode測(cè)試')

# 或者使用接收二進(jìn)制編碼的命令

with open('Utf8.txt', 'wb') as f: f.write('Utf8測(cè)試')

# 你可以反過(guò)來(lái)做個(gè)測(cè)試,自然會(huì)報(bào)錯(cuò)

# 二進(jìn)制的命令方便了在不知道怎么解碼的情況下也能進(jìn)行操作(寫(xiě)入文件)

三. 建議的使用習(xí)慣

相信到這里我已經(jīng)把我對(duì)于編碼的理解講完了。

我們?yōu)槭裁磿?huì)碰到各式各樣的編碼問(wèn)題:

? 因?yàn)槲覀儧](méi)有統(tǒng)一編碼

? 因?yàn)槲覀儧](méi)有用對(duì)命令(傳對(duì)數(shù)據(jù))

所以這里再重申一下八字真言:確定編碼,同類(lèi)交互

? 碰到問(wèn)題,問(wèn)一下自己,我現(xiàn)在是哪種編碼

? 同一種編碼才能交互,那我應(yīng)該是哪種編碼

這里給出我的使用習(xí)慣:

? 確定一種內(nèi)部編碼

? 內(nèi)部編碼的選擇優(yōu)先級(jí)如下:程序必須使用的編碼、第三方包使用的編碼、你喜歡的編碼、Unicode

? 在輸出時(shí)再更改到特定的編碼

記得在開(kāi)始整個(gè)程序之前確定內(nèi)部的編碼,否則編碼一團(tuán)糟會(huì)產(chǎn)生很多不必要的bug。

不要迷信內(nèi)部Unicode,例如Evernote開(kāi)發(fā)就應(yīng)該根據(jù)第三方包使用的Utf8確定內(nèi)部編碼。

四. 疑難問(wèn)題解答

編碼識(shí)別

說(shuō)了要確定編碼,那么拿到一串二進(jìn)制要怎么確定編碼呢?

最簡(jiǎn)單的方法是chardet:(需要安裝)

python -m pip install chardet

使用非常簡(jiǎn)單:

#coding=utf8

from chardet import detect

print(detect('這是一串utf8的測(cè)試字符'))

# 結(jié)果:`{'confidence': 0.99, 'encoding': 'utf-8'}`

另外例如抓取網(wǎng)站,那么頭文件中很有可能有提示如何解碼,記得不要忘記了。

編碼轉(zhuǎn)換

很可能因?yàn)樽址袇㈦s了奇怪的東西,導(dǎo)致即使編碼種類(lèi)正確,依舊無(wú)法解碼。

我知道我之前講過(guò)了,但可能有人直接跳疑難問(wèn)題解答嘛。

這里可以使用decode的第二個(gè)參數(shù):

#coding=utf8

# 字符串中混進(jìn)了\x00

rubbishUtf8String = 'Utf-8字\x00符串'

print(repr(rubbishUtf8String.decode('utf8', 'replace')))

print(repr(rubbishUtf8String.decode('utf8', 'ignore')))

特殊平臺(tái)下編碼

很多人都說(shuō)Windows是個(gè)坑,即使在Python 3下面也一樣。

因?yàn)橹形奈募鰜?lái)都是亂碼。

這里使用一個(gè)取巧的方法:平臺(tái)編碼再特殊,起碼命令行讀取和創(chuàng)建一個(gè)文件夾不會(huì)出亂碼吧。

import sys, os

for folder in os.walk('.').next()[1]:

print(folder.decode(sys.stdin.encoding))

同樣的輸入輸出也可以這樣做優(yōu)化:

import sys

def sys_print(msg):

print(msg.encode(sys.stdin.encoding))

def sys_input(msg):

return raw_input(msg.encode(sys.stdin.encoding)).decode(sys.stdin.encoding)

文件寫(xiě)入

如果抓下來(lái)一個(gè)內(nèi)容不知道怎么解碼,但還是想要寫(xiě)入文件怎么辦

寫(xiě)入文件的時(shí)候制定用二進(jìn)制命令即可:

#coding=utf8

import urllib

with open('Utf8.txt', 'wb') as f: f.write('Utf8測(cè)試')

# 比如抓了個(gè)網(wǎng)頁(yè),不知道編碼也可以寫(xiě)入文件進(jìn)行一系列操作

content = urllib.urlopen('http://www.baidu.com').read()

with open('baidu.txt', 'wb') as f: f.write(content)

裸Unicode字符

Unicode存成六個(gè)Ascii字符怎么辦?其實(shí)也可以decode

#coding=utf8

# 這是普通的Unicode

s = u'測(cè)'

for i in s: print(i)

print(repr(s))

# 這是裸Unicode,實(shí)際存成了六個(gè)Ascii

s = repr(s)[2:-1]

for i in s: print(i)

print(repr(s))

# 轉(zhuǎn)化其實(shí)也很簡(jiǎn)單

s = s.decode('unicode-escape')

for i in s: print(i)

print(repr(s))

好了,希望這篇文章對(duì)大家解決Python編碼問(wèn)題有所幫助!【注:需要更多免費(fèi)學(xué)習(xí)視頻+資料+源碼,請(qǐng)加QQ:2632311208】


本文版權(quán)歸黑馬程序員人工智能+Python學(xué)院所有,歡迎轉(zhuǎn)載,轉(zhuǎn)載請(qǐng)注明作者出處。謝謝!


作者:黑馬程序員人工智能+Python培訓(xùn)學(xué)院


首發(fā):http://python.itheima.com/


分享到:
在線咨詢(xún) 我要報(bào)名
和我們?cè)诰€交談!