<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    jsp亂碼問題剖析與終極解決方案

    Posted on 2009-12-17 17:20 terryxue 閱讀(1542) 評論(0)  編輯  收藏 所屬分類: java
    服務器返回亂碼頁面,請求的數據發送到服務器后取出來是亂碼,以上兩個問題是web開發人員經常遇到的問題,解決這類問題需要理解亂碼問題的根源所在。

    1. 字符編碼
    字符是以二進制編碼的形式保存在存儲器中的,如:“我”這個字,可以用gbk的方式保存(用字節表示是[-50, -46]),也可以用utf-8的方式保存(用字節表示是[-26, -120, -111])。程序在讀取數據塊時需要一個字節一個字節的讀取,然后將字節轉換為字符,顯然如果程序不知道字節是表示的什么編碼的字符,讀出來就會出問 題,這就如同你說了一句話,我如果事先不知道你說的將是什么語言,就沒法去翻譯了,如果你說的英語,我以為是日語,然后按日語的方式來理解到我的主觀意 識,顯然結果就是不知你在說什么了,就成了我們程序中的亂碼。所以對于前面我所表示的字節數組byte[] data = {-26, -120, -111},我們必須這樣做才能得到正確的字符: String s = new String(data, "utf-8"),(注:這里用了字符串,因為字符串就是由一個個字符組成的),如果我們不指定參數里面的"utf-8",那么系統就會用操作系統默認的 編碼了,這可能是gbk或是什么任何編碼。

    2. jsp服務器返回亂碼頁面
      2.1 pageEncoding
      在jsp頁面的page指令中我們指定了pageEncoding屬性,這個屬性就是告訴jsp容器如何讀取這個jsp頁面,所以這個屬性必須與jsp頁 面保存的編碼保持一致。也就是說,如果你頁面的編碼保存為gbk, 而pageEncoding設置成了utf-8,則jsp容器在讀這個jsp頁面的時候就會出錯(如果存在非英文字符的話)。jsp容器讀jsp的目的是 將其翻譯成java代碼,所以如果讀錯了jsp頁面,翻譯出來的java代碼也就會出錯,如果這種錯誤影響了java文件的語法,就會在訪問時出現無法編 譯jsp的語法錯誤,如果沒有影響到語法,就會出現最終顯示的html頁面上有亂碼的錯誤。所以如果遇到顯示亂碼,則檢查pageEncoding是否正 確。
      2.2 contentType
      page指令中的contentType屬性用于指定返回給瀏覽器的數據的文檔類型,服務器通過http頭信息返回給瀏覽器這個信息,所以在瀏覽器 html代碼中用戶是看不到的。同時contentType屬性還可以指定頁面的編碼,即服務器即以什么編碼發送頁面數據。比如說中文數據,可以用gbk 或utf-8的方式來發送,這個編碼跟jsp頁面的編碼沒有關系,只要設定的編碼支持頁面中的字符就行了。相同于有了一個字符串s="中國人",然后用 s.getBytes("gbk")的方式來發送s。所以由于contentType錯誤出現亂碼的概念不高,但也要注意一下,比如說如果設置成 了"iso8859-1",則瀏覽器就會顯示亂碼了。contentType還有一個用,就是瀏覽器將會依據這個編碼來顯示頁面,在IE下點右鍵,然后選 擇“編碼”,你就可以注意到頁面是以什么編碼顯示的了。
      2.3 如果是servlet返回的結果
      上面說的是jsp,如果servlet的話就要注意設置response.setCharacterEncoding(""),如果沒有設置,服務器會默 認為是iso8859-1,設置后得到的writer(即response.getWriter())對象,就會依據這個編碼來向客戶端寫數 據,writer對象的構造與以下方式類似:PrintWriter pw = new PrinterWriter(new OutputStreamWriter(socket.getOutputStream(), "編碼")),這里提到了通過socket得到輸出流,不明白的話可以參考我的另一篇文章。pw.write("你好"),實際上就是先通過byte[] data = "你好".getBytes("編碼"),然后將data寫給客戶端。
    )

    3. 服務器得到客戶端傳過來的數據為亂碼
      3.1 通用解決方案
      String param = request.getParameter("paramName"),如果瀏覽器傳過來的為中文,則取出來的數據將是亂碼。為什么呢?因為客戶端只能將 數據的編碼傳給服務器,如[-26, -120, -111],但服務器并不知道這是什么字符集的編碼,于是假定為iso8859-1, 用這種方式構造了字符串s = new String(data, "iso8859-1"),顯然這樣肯定是亂碼。解決方案很簡單,我們得到值param后,用byte data[] = param.getBytes("iso8859-1"),這樣data就是客戶端傳過來的真實編碼,然后我們再重新創建字符串:param = new String(data, "正確的編碼");
      3.2 POST請求
      如于post請求處理起來更簡單,get請求與post請求向服務器發送數據的方式不一樣,get請求的參數是通過HTTP頭信息中的第一行數據發送的, 是URI的一部分,而post請求則是在發送完HTTP頭信息后作為單獨的數據塊發送的。因此對于get請求的參數,我們在使用request之前服務器 已經讀出來了,已經是亂碼了,只能用前面的方案,但對于post請求的數據,我們在調用getParameter或getReader之前,服務器并沒有 去處理,所以我們可以在getParameter之前先告訴服務器正確的編碼,通過request.setCharacterEncoding("正確的 編碼"),然后再讀取參數。

    4 AJax

      對于ajax請求注意要用utf-8編碼,request和response都需要使用utf-8

    posts - 9, comments - 24, trackbacks - 0, articles - 0

    Copyright © terryxue

    主站蜘蛛池模板: 亚洲成a人片在线不卡一二三区 | 免费人成在线视频| 亚洲精品中文字幕乱码影院| 无码少妇精品一区二区免费动态 | 久久精品国产精品亚洲艾草网 | 国产麻豆视频免费观看| 亚洲天堂一区二区三区四区| 99在线精品免费视频九九视 | 国产精品亚洲专区无码唯爱网| 日韩免费福利视频| 免费大片av手机看片高清| 亚洲精品视频免费观看| 成年网在线观看免费观看网址| 久久久久亚洲精品天堂久久久久久| xxxx日本在线播放免费不卡| 国产aⅴ无码专区亚洲av麻豆| 成人电影在线免费观看| 亚洲最大的视频网站| 午夜私人影院免费体验区| 国产亚洲精品2021自在线| 亚洲日韩中文在线精品第一| 日本亚洲欧洲免费天堂午夜看片女人员 | 综合久久久久久中文字幕亚洲国产国产综合一区首 | 99蜜桃在线观看免费视频网站| 亚洲精品偷拍无码不卡av| 免费黄色一级毛片| 一级毛片免费毛片毛片| 久久青青草原亚洲av无码app | 亚洲一级二级三级不卡| 国产va免费精品观看精品| 国产精品亚洲专区在线播放| 亚洲精品无码MV在线观看| 在线看片v免费观看视频777| 精品国产亚洲一区二区三区在线观看| avtt亚洲天堂| 一级毛片在线免费观看| 国产精品久久亚洲一区二区| 亚洲av午夜成人片精品网站| 毛片免费全部播放一级| 韩日电影在线播放免费版| 亚洲人成网站色在线观看|