寫python腳本的時候發(fā)現(xiàn)這樣一個問題:從xls文件導(dǎo)出到txt時,無法直接轉(zhuǎn)換為int型數(shù)據(jù),輸出查看發(fā)現(xiàn)和文件編碼方式產(chǎn)生的附加信息有關(guān)用一個簡單的文件舉例 90905 90907 90908 90909 90939 90940 90946 90959 90961 90965 當(dāng)文件分別用ascii,utf8,utf8 bom作為編碼格式時,顯示輸出結(jié)果如下: 使用ascii編碼的輸出: ['90905\r\n', '90907\r\n', '90908\r\n', '90909\r\n', '90939\r\n', '90940\r\n', '90946\r\n', '90959\r\n', '90961\r\n', '90965'] 使用utf8編碼的輸出: ['90905\r\n', '90907\r\n', '90908\r\n', '90909\r\n', '90939\r\n', '90940\r\n', '90946\r\n', '90959\r\n', '90961\r\n', '90965'] 使用bom編碼的輸出: ['\xef\xbb\xbf90905\r\n', '90907\r\n', '90908\r\n', '90909\r\n', '90939\r\n', '90940\r\n', '90946\r\n', '90959\r\n', '90961\r\n', '90965'] 原來utf8 bom不能直接轉(zhuǎn)換int的原因在這里,它在文件頭插入了一個表示文件編碼的信息\xef\xbb\xbf,那么UTF-8(無BOM)和UTF-8這兩個有什么區(qū)別呢?BOM是什么呢? 什么是BOM?BOM: Byte Order MarkUTF-8 BOM又叫UTF-8 簽名,其實UTF-8 的BOM對UFT-8沒有作用,是為了支持UTF-16,UTF-32才加上的 BOM,BOM簽名的意思就是告訴編輯器當(dāng)前文件采用何種編碼,方便編輯器識別,但是BOM雖然在編輯器中不顯示,但是會產(chǎn)生輸出,就像多了一個空行。 Byte Order Marks are special characters at the beginning of a Unicode file to indicate whether it is big or little endian, in other words does the high or low order byte come first. These codes also tell whether the encoding is 8, 16 or 32 bit. You can recognise Unicode files by their starting byte order marks, and by the way Unicode-16 files are half zeroes and Unicode-32 files are three-quarters zeros. Unicode Endian Markers Byte-order mark Description UTF的字節(jié)序和BOMUTF-8以字節(jié)為編碼單元,沒有字節(jié)序的問題。UTF-16以兩個字節(jié)為編碼單元,在解釋一個UTF-16文本前,首先要弄清楚每個編碼單元的字節(jié)序。例如收到一個“奎”的Unicode編碼是594E,“乙”的Unicode編碼是4E59。如果我們收到UTF-16字節(jié)流“594E”,那么這是“奎”還是“乙”? Unicode規(guī)范中推薦的標(biāo)記字節(jié)順序的方法是BOM。BOM不是“Bill Of Material”的BOM表,而是Byte Order Mark。BOM是一個有點小聰明的想法: 在UCS編碼中有一個叫做'ZERO WIDTH NO-BREAK SPACE'的字符,它的編碼是FEFF。而FFFE在UCS中是不存在的字符,所以不應(yīng)該出現(xiàn)在實際傳輸中。UCS規(guī)范建議我們在傳輸字節(jié)流前,先傳輸字符'ZERO WIDTH NO-BREAK SPACE'。 這樣如果接收者收到FEFF,就表明這個字節(jié)流是Big-Endian的;如果收到FFFE,就表明這個字節(jié)流是Little-Endian的。因此字符'ZERO WIDTH NO-BREAK SPACE'又被稱作BOM。 UTF-8不需要BOM來表明字節(jié)順序,但可以用BOM來表明編碼方式。字符'ZERO WIDTH NO-BREAK SPACE'的UTF-8編碼是EF BB BF。所以如果接收者收到以EF BB BF開頭的字節(jié)流,就知道這是UTF-8編碼了。 Windows就是使用BOM來標(biāo)記文本文件的編碼方式的。 原來BOM是在文件的開始加了幾個字節(jié)作為標(biāo)記。有了這個標(biāo)記,一些協(xié)議和系統(tǒng)才能識別。 ok,說了這么多背景,那么如何解決這個問題呢? 如何使用BOM頭BOM頭的刪除對UTF-16, Python將BOM解碼為空字串。然而對UTF-8, BOM被解碼為一個字符,如例: >>> codecs.BOM_UTF16.decode( 'utf16' ) u'' >>> codecs.BOM_UTF8.decode( 'utf8' ) u'\ufeff'
簡單的做法是在文件讀入時使用 import codecs
f = codecs.open(sys.argv[1],'r', 'utf_8_sig')
即可,具體可以參見[http://docs./library/codecs.html#module-encodings.utf_8_sig|http://docs./library/codecs.html#module-encodings.utf_8_sig] 或者: u.lstrip( unicode( codecs.BOM_UTF8, 'utf8' ) )
BOM頭的添加out = file( 'someFile', 'w' )
out.write( codecs.BOM_UTF8 )
out.write( unicodeString.encode( 'utf-8' ) )
out.close()out = file( 'someFile', 'w' )out.write( codecs.BOM_UTF8 )out.write( unicodeString.encode( 'utf-8' ) )out.close()
詳細(xì)的過程解釋可以參見[http:///jgloss/encoding.html|http:///jgloss/encoding.html] 參考資料: [http://blog.sina.com.cn/s/blog_3e9d2b350100as0b.html|http://blog.sina.com.cn/s/blog_3e9d2b350100as0b.html] [http://4nail./blog/840612|http://4nail./blog/840612] |
|