亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JSP > 正文

JSP/Servlet中的漢字編碼問題

2024-09-05 00:20:00
字體:
來源:轉載
供稿:網友

網上就 jsp/servlet 中 dbcs 字符編碼問題有許多優秀的文章和討論,本文對它們作一些整理,并結合 ibm websphere application server 3.5(was)的解決方法作一些說明,希望它不是多余的。
內容:

問題的起源
gb2312-80,gbk,gb18030-2000 漢字字符集及 encoding
中文轉碼時´?´、亂碼的由來
jsp/servlet 漢字編碼問題及在 was 中的解決辦法
結束語
參考文章

1. 問題的起源

每個國家(或區域)都規定了計算機信息交換用的字符編碼集,如美國的擴展 ascii碼, 中國的 gb2312-80,日本的 jis 等,作為該國家/區域內信息處理的基礎,有著統一編碼的重要作用。字符編碼集按長度分為 sbcs(單字節字符集),dbcs(雙字節字符集)兩大類。早期的軟件(尤其是操作系統),為了解決本地字符信息的計算機處理,出現了各種本地化版本(l10n),為了區分,引進了 lang, codepage 等概念。但是由于各個本地字符集代碼范圍重疊,相互間信息交換困難;軟件各個本地化版本獨立維護成本較高。因此有必要將本地化工作中的共性抽取出來,作一致處理,將特別的本地化處理內容降低到最少。這也就是所謂的國際化(i18n)。各種語言信息被進一步規范為 locale 信息。處理的底層字符集變成了幾乎包含了所有字形的 unicode。

現在大部分具有國際化特征的軟件核心字符處理都是以 unicode 為基礎的,在軟件運行時根據當時的 locale/lang/codepage 設置確定相應的本地字符編碼設置,并依此處理本地字符。在處理過程中需要實現 unicode 和本地字符集的相互轉換,甚或以 unicode 為中間的兩個不同本地字符集的相互轉換。這種方式在網絡環境下被進一步延伸,任何網絡兩端的字符信息也需要根據字符集的設置轉換成可接受的內容。

java 語言內部是用 unicode 表示字符的,遵守 unicode v2.0。java 程序無論是從/往文件系統以字符流讀/寫文件,還是往 url 連接寫 html 信息,或從 url 連接讀取參數值,都會有字符編碼的轉換。這樣做雖然增加了編程的復雜度,容易引起混淆,但卻是符合國際化的思想的。

從理論上來說,這些根據字符集設置而進行的字符轉換不應該產生太多問題。而事實是由于應用程序的實際運行環境不同,unicode 和各個本地字符集的補充、完善,以及系統或應用程序實現的不規范,轉碼時出現的問題時時困擾著程序員和用戶。

2. gb2312-80,gbk,gb18030-2000 漢字字符集及 encoding

其實解決 java 程序中的漢字編碼問題的方法往往很簡單,但理解其背后的原因,定位問題,還需要了解現有的漢字編碼和編碼轉換。

gb2312-80 是在國內計算機漢字信息技術發展初始階段制定的,其中包含了大部分常用的一、二級漢字,和 9 區的符號。該字符集是幾乎所有的中文系統和國際化的軟件都支持的中文字符集,這也是最基本的中文字符集。其編碼范圍是高位0xa1-0xfe,低位也是 0xa1-0xfe;漢字從 0xb0a1 開始,結束于 0xf7fe;

gbk 是 gb2312-80 的擴展,是向上兼容的。它包含了 20902 個漢字,其編碼范圍是 0x8140-0xfefe,剔除高位 0x80 的字位。其所有字符都可以一對一映射到 unicode 2.0,也就是說 java 實際上提供了 gbk 字符集的支持。這是現階段 windows 和其它一些中文操作系統的缺省字符集,但并不是所有的國際化軟件都支持該字符集,感覺是他們并不完全知道 gbk 是怎么回事。值得注意的是它不是國家標準,而只是規范。隨著 gb18030-2000國標的發布,它將在不久的將來完成它的歷史使命。

gb18030-2000(gbk2k) 在 gbk 的基礎上進一步擴展了漢字,增加了藏、蒙等少數民族的字形。gbk2k 從根本上解決了字位不夠,字形不足的問題。它有幾個特點,

它并沒有確定所有的字形,只是規定了編碼范圍,留待以后擴充。
編碼是變長的,其二字節部分與 gbk 兼容;四字節部分是擴充的字形、字位,其編碼范圍是首字節 0x81-0xfe、二字節0x30-0x39、三字節 0x81-0xfe、四字節0x30-0x39。
它的推廣是分階段的,首先要求實現的是能夠完全映射到 unicode 3.0 標準的所有字形。
它是國家標準,是強制性的。
現在還沒有任何一個操作系統或軟件實現了 gbk2k 的支持,這是現階段和將來漢化的工作內容。
unicode 的介紹......就免了吧。

java 支持的encoding中與中文編程相關的有:(有幾個在jdk文檔中未列出)

ascii 7-bit, 同 ascii7
iso8859-1 8-bit, 同 8859_1,iso-8859-1,iso_8859-1,latin1...
gb2312-80 同gb2312,gb2312-1980,euc_cn,euccn,1381,cp1381, 1383, cp1383, iso2022cn,iso2022cn_gb......
gbk (注意大小寫),同ms936
utf8 utf-8
gb18030 (現在只有ibm jdk1.3.?有支持), 同cp1392,1392

java 語言采用unicode處理字符. 但從另一個角度來說,在java程序中也可以采用非unicode的轉碼,重要的是保證程序入口和出口的漢字信息不失真。如完全采用iso-8859-1來處理漢字也能達到正確的結果。網絡上流行的許多解決方法,都屬于這種類型。為了不致引起混淆,本文不對這種方法作討論。

3. 中文轉碼時´?´、亂碼的由來

兩個方向轉換都有可能得到錯誤的結果:

unicode-->byte, 如果目標代碼集不存在對應的代碼,則得到的結果是0x3f.
如:
"u00d6u00ecu00e9u0046u00bbu00f9".getbytes("gbk") 的結果是 "?ìéf?ù", hex 值是3fa8aca8a6463fa8b4.

仔細看一下上面的結果,你會發現u00ec被轉換為0xa8ac, u00e9被轉換為xa8a6... 它的實際有效位變長了!這是因為gb2312符號區中的一些符號被映射到一些公共的符號編碼,由于這些符號出現在iso-8859-1或其它一些sbcs字符集中,故它們在unicode中編碼比較靠前,有一些其有效位只有8位,和漢字的編碼重疊(其實這種映射只是編碼的映射,在顯示時仔細不是一樣的。unicode 中的符號是單字節寬,漢字中的符號是雙字節寬) . 在unicodeu00a0--u00ff 之間這樣的符號有20個。了解這個特征非常重要!由此就不難理解為什么java編程中,漢字編碼的錯誤結果中常常會出現一些亂碼(其實是符號字符), 而不全是´?´字符, 就比如上面的例子。

byte-->unicode, 如果byte標識的字符在源代碼集不存在,則得到的結果是0xfffd.
如:
byte ba[] = {(byte)0x81,(byte)0x40,(byte)0xb0,(byte)0xa1}; new string(ba,"gb2312");
結果是"?啊", hex 值是"ufffdu554a". 0x8140 是gbk字符,按gb2312轉換表沒有對應的值,取ufffd. (請注意:在顯示該unicode時,因為沒有對應的本地字符,所以也適用上一種情況,顯示為一個"?".)

實際編程中,jsp/servlet 程序得到錯誤的漢字信息,往往是這兩個過程的疊加,有時甚至是兩個過程疊加后反復作用的結果.

4. jsp/servlet 漢字編碼問題及在 was 中的解決辦法

4.1 常見的 encoding 問題的現象
網上常出現的 jsp/servlet encoding 問題一般都表現在 browser 或應用程序端,如:
瀏覽器中看到的 jsp/servlet 頁面中的漢字怎么都成了 ’?’ ?
瀏覽器中看到的 servlet 頁面中的漢字怎么都成了亂碼?
java 應用程序界面中的漢字怎么都成了方塊?
jsp/servlet 頁面無法顯示 gbk 漢字。
jsp 頁面中內嵌在<%...%>,<%=...%>等tag包含的 java code 中的中文成了亂碼,但頁面的其它漢字是對的。
jsp/servlet 不能接收 form 提交的漢字。
jsp/servlet 數據庫讀寫無法獲得正確的內容。
隱藏在這些問題后面的是各種錯誤的字符轉換和處理(除第3個外,是因為 java font 設置錯誤引起的)。解決類似的字符 encoding 問題,需要了解 jsp/servlet 的運行過程,檢查可能出現問題的各個點。

4.2 jsp/servlet web 編程時的 encoding 問題
運行于java 應用服務器的 jsp/servlet 為 browser 提供 html 內容,其過程如下圖所示:

其中有字符編碼轉換的地方有:

jsp 編譯。java 應用服務器將根據 jvm 的 file.encoding 值讀取 jsp 源文件,編譯生成 java 源文件,再根據 file.encoding 值寫回文件系統。如果當前系統語言支持 gbk,那么這時候不會出現 encoding 問題。如果是英文的系統,如 lang 是 en_us 的 linux, aix 或 solaris,則要將 jvm 的 file.encoding 值置成 gbk 。系統語言如果是 gb2312,則根據需要,確定要不要設置 file.encoding,將 file.encoding 設為 gbk 可以解決潛在的 gbk 字符亂碼問題

java 需要被編譯為 .class 才能在 jvm 中執行,這個過程存在與a.同樣的 file.encoding 問題。從這里開始 servlet 和 jsp 的運行就類似了,只不過 servlet 的編譯不是自動進行的。對于jsp程序, 對產生的java 中間文件的編譯是自動進行的(在程序中直接調用sun.tools.javac.main類). 因此如果在這一步出現問題的話, 也要檢查encoding和os的語言環境,或者將內嵌在jsp java code 中的靜態漢字轉為 unicode, 要么靜態文本輸出不要放在 java code 中。對于servlet, javac 編譯時手工指定-encoding 參數就可以了。

servlet 需要將 html 頁面內容轉換為 browser 可接受的 encoding 內容發送出去。依賴于各 java app server 的實現方式,有的將查詢 browser 的 accept-charset 和 accept-language 參數或以其它猜的方式確定 encoding 值,有的則不管。因此采用固定encoding 也許是最好的解決方法。對于中文網頁,可在 jsp 或 servlet 中設置 contenttype="text/html; charset=gb2312";如果頁面中有gbk字符,則設置為contenttype="text/html; charset=gbk",由于ie 和 netscape對gbk的支持程度不一樣,作這種設置時需要測試一下。
因為16位 java char在網絡傳送時高8位會被丟棄,也為了確保servlet頁面中的漢字(包括內嵌的和servlet運行過程中得到的)是期望的內碼,可以用 printwriter out=res.getwriter() 取代 servletoutputstream out=res.getoutputstream(). printerwriter 將根據contenttype中指定的charset作轉換 (contenttype需在此之前指定!); 也可以用outputstreamwriter封裝 servletoutputstream 類并用write(string)輸出漢字字符串。
對于 jsp,java application server 應當能夠確保在這個階段將嵌入的漢字正確傳送出去。

這是解釋 url 字符 encoding 問題。如果通過 get/post 方式從 browser 返回的參數值中包含漢字信息, servlet 將無法得到正確的值。sun的 j2sdk 中,httputils.parsename 在解析參數時根本沒有考慮 browser 的語言設置,而是將得到的值按 byte 方式解析。這是網上討論得最多的 encoding 問題。因為這是設計缺陷,只能以 bin 方式重新解析得到的字符串;或者以 hack httputils 類的方式解決。參考文章 2 均有介紹,不過最好將其中的中文 encoding gb2312、 cp1381 都改為 gbk,否則遇到 gbk 漢字時,還是會有問題。
servlet api 2.3 提供一個新的函數 httpserveletrequest.setcharacterencoding 用于在調用 request.getparameter(“param_name”) 前指定應用程序希望的 encoding,這將有助于徹底解決這個問題。
4.3 ibm websphere application server 中的解決方法

websphere application server 對標準的 servlet api 2.x 作了擴展,提供較好的多語言支持。運行在中文的操作系統中,可以不作任何設置就可以很好地處理漢字。下面的說明只是對was是運行在英文的系統中,或者需要有gbk支持時有效。

上述c,d情況,was 都要查詢 browser 的語言設置,在缺省狀況下, zh, zh-cn 等均被映射為 java encoding cp1381(注意: cp1381 只是等同于 gb2312 的一個 codepage,沒有 gbk 支持)。這樣做我想是因為無法確認 browser 運行的操作系統是支持gb2312, 還是 gbk,所以取其小。但是實際的應用系統還是要求頁面中出現 gbk 漢字,最著名的是朱總理名字中的“?"(rong2 ,0xe946,u9555),所以有時還是需要將 encoding/charset 指定為 gbk。當然 was 中變更缺省的 encoding 沒有上面說的那么麻煩,針對 a,b,參考文章 5,在 application server 的命令行參數中指定 -dfile.encoding=gbk 即可; 針對 d,在 application server 的命令行參數中指定-ddefault.client.encoding=gbk。如果指定了-ddefault.client.encoding=gbk,那么c情況下可以不再指定charset。

上面列出的問題中還有一個關于tag<%...%>,<%=...%>中的 java 代碼里包含的靜態文本未能正確顯示的問題,在was中的解決方法是除了設置正確的file.encoding, 還需要以相同方法設置-duser.language=zh -duser.region=cn。這與java locale的設置有關。

4.4 數據庫讀寫時的 encoding 問題

jsp/servlet 編程中經常出現 encoding 問題的另一個地方是讀寫數據庫中的數據。

流行的關系數據庫系統都支持數據庫 encoding,也就是說在創建數據庫時可以指定它自己的字符集設置,數據庫的數據以指定的編碼形式存儲。當應用程序訪問數據時,在入口和出口處都會有 encoding 轉換。對于中文數據,數據庫字符編碼的設置應當保證數據的完整性. gb2312,gbk,utf-8 等都是可選的數據庫 encoding;也可以選擇 iso8859-1 (8-bit),那么應用程序在寫數據之前須將 16bit 的一個漢字或 unicode 拆分成兩個 8-bit 的字符,讀數據之后則需將兩個字節合并起來,同時還要判別其中的 sbcs 字符。沒有充分利用數據庫 encoding 的作用,反而增加了編程的復雜度,iso8859-1不是推薦的數據庫 encoding。jsp/servlet編程時,可以先用數據庫管理系統提供的管理功能檢查其中的中文數據是否正確。

然后應當注意的是讀出來的數據的 encoding,java 程序中一般得到的是 unicode。寫數據時則相反。

4.5 定位問題時常用的技巧

定位中文encoding問題通常采用最笨的也是最有效的辦法??在你認為有嫌疑的程序處理后打印字符串的內碼。通過打印字符串的內碼,你可以發現什么時候中文字符被轉換成unicode,什么時候unicode被轉回中文內碼,什么時候一個中文字成了兩個 unicode 字符,什么時候中文字符串被轉成了一串問號,什么時候中文字符串的高位被截掉了……

取用合適的樣本字符串也有助于區分問題的類型。如:”aa啊aa?aa” 等中英相間、gb、gbk特征字符均有的字符串。一般來說,英文字符無論怎么轉換或處理,都不會失真(如果遇到了,可以嘗試著增加連續的英文字母長度)。

5. 結束語

其實 jsp/servlet 的中文encoding 并沒有想像的那么復雜,雖然定位和解決問題沒有定規,各種運行環境也各不盡然,但后面的原理是一樣的。了解字符集的知識是解決字符問題的基礎。不過,隨著中文字符集的變化,不僅僅是 java 編程,中文信息處理中的問題還是會存在一段時間的。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
2019亚洲男人天堂| 亚洲人精品午夜在线观看| 萌白酱国产一区二区| 国产热re99久久6国产精品| 日韩av一区二区在线观看| 欧美日韩一区二区免费视频| 欧美激情免费看| 亚洲乱码国产乱码精品精| 成人激情在线播放| 久久久av网站| 欧美亚洲视频一区二区| 欧美精品福利在线| 成人免费视频xnxx.com| 亚洲三级 欧美三级| 成人国内精品久久久久一区| 成人免费视频在线观看超级碰| 欧美性xxxx极品hd欧美风情| 九九热这里只有在线精品视| 久久久成人的性感天堂| 91色精品视频在线| 欧美老女人在线视频| 国模视频一区二区| 亚洲精品美女在线| 欧美综合在线第二页| 亚洲一区二区三区xxx视频| 亚洲3p在线观看| 日本免费在线精品| 久久激情视频久久| 免费91麻豆精品国产自产在线观看| 国内久久久精品| 欧美高清不卡在线| 日韩一区二区三区xxxx| 国产精品第一区| 色中色综合影院手机版在线观看| 亚洲视频在线观看免费| 国产精品免费小视频| 免费91麻豆精品国产自产在线观看| 1769国产精品| 久久电影一区二区| 国产999精品久久久影片官网| 美日韩丰满少妇在线观看| 亚洲人免费视频| 中文字幕日韩精品在线| 日本欧美在线视频| 日韩av大片免费看| 午夜美女久久久久爽久久| 国产精品自拍偷拍视频| 亚洲国内高清视频| 最新日韩中文字幕| 中文字幕亚洲无线码在线一区| 91精品国产91| 成人动漫网站在线观看| 精品久久久久久久久久ntr影视| 国产精品扒开腿做爽爽爽视频| 欧美精品在线免费观看| 国产欧美精品xxxx另类| 亚洲国产成人爱av在线播放| 日韩av免费看网站| 精品国产网站地址| 亚洲电影免费观看高清完整版在线| 欧美影院久久久| 美女av一区二区| 久久精品电影网| 国产精品专区第二| 日韩视频免费在线观看| 亚洲男人天堂网站| 精品国产一区二区三区久久| 久久成人在线视频| 国产一区二区三区网站| 亚洲变态欧美另类捆绑| 欧美国产日韩在线| 欧美日韩激情美女| 91亚洲人电影| 这里只有精品视频在线| 亚洲精品国产精品国产自| 北条麻妃一区二区三区中文字幕| 97成人精品视频在线观看| 欧美日韩国产成人在线观看| 欧美日韩国产精品一区二区不卡中文| 激情成人在线视频| 欧美激情视频网| 亚洲欧美国产精品| 欧美成人在线免费| 欧美中文在线观看| 欧美日韩激情小视频| 欧美电影免费观看大全| 国产综合色香蕉精品| 亚洲精品视频免费| 在线一区二区日韩| 国语自产精品视频在线看| 精品福利视频导航| 国产欧美日韩亚洲精品| 久久久久久久久网站| 中文字幕最新精品| 国产精品最新在线观看| 国产成人精品视频| 久久精品久久久久| 亚洲一级一级97网| 美女福利精品视频| 影音先锋欧美在线资源| 欧美激情亚洲另类| 精品夜色国产国偷在线| 视频在线观看一区二区| 亚洲乱码国产乱码精品精天堂| 国产精品一区二区三区在线播放| 热re99久久精品国产66热| 一区三区二区视频| 欧美成人免费一级人片100| 国产午夜精品全部视频播放| 亚洲免费精彩视频| 久久久久九九九九| 成人免费视频xnxx.com| 色久欧美在线视频观看| 欧美精品videosex牲欧美| 2019最新中文字幕| 96pao国产成视频永久免费| 久久久精品国产亚洲| 日韩电影免费在线观看| 国产视频在线观看一区二区| 18久久久久久| 97在线免费观看视频| 欧美黄色片在线观看| 亚洲精品国产精品国自产观看浪潮| 亚洲综合小说区| 日韩在线观看免费网站| 国产精品白嫩初高中害羞小美女| 色偷偷88888欧美精品久久久| 国语自产精品视频在免费| 久久综合伊人77777蜜臀| 最近2019年日本中文免费字幕| 成人美女免费网站视频| 国产精品成人播放| 日韩精品久久久久久久玫瑰园| 国产精品999999| 色综合久综合久久综合久鬼88| 92版电视剧仙鹤神针在线观看| 中文字幕精品www乱入免费视频| 国外色69视频在线观看| 91热精品视频| 欧美日韩国产999| 亚洲免费一在线| 人体精品一二三区| 国产精品尤物福利片在线观看| 国产视频一区在线| 国产精品入口免费视| 欧美国产日产韩国视频| 亚洲精品国产电影| 热草久综合在线| 久久国产精品电影| www.欧美三级电影.com| 国产亚洲精品久久久久动| 久久久999精品| 国产亚洲精品美女久久久久| 国产精自产拍久久久久久| 91成人免费观看网站| 欧美日韩国产在线播放| www亚洲精品| 精品国产电影一区| 国产日韩欧美在线观看| 亚洲国产日韩欧美在线动漫| 日韩在线视频线视频免费网站| 欧美孕妇与黑人孕交| 亚洲视频第一页| 成人午夜在线观看|