【轉】Ajax響應中文亂碼 [SpringMVC使用@ResponseBody處理Ajax請求]

Spring3.0 MVC @ResponseBody 的做用是把返回值直接寫到HTTP response body裏。  html

 

Spring使用AnnotationMethodHandlerAdapter的handleResponseBody方法, AnnotationMethodHandlerAdapter使用request header中"Accept"的值和messageConverter支持的MediaType進行匹配,而後會用"Accept"的第一個值寫入response的"Content-Type"。
通常的請求都是經過瀏覽器進行的,request header中"Accept"的值由瀏覽器生成。

  

有人跟蹤@ResponseBody 的實現類發現其默認的編碼是 iso-8859-1, java

解決辦法,在spring mvc的配置文件中手工配置bean:
<!-- 啓動Spring MVC的註解功能,完成請求和註解POJO的映射 -->    
<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter" >  
<property name="messageConverters">   
         <list>   
             <bean class = "org.springframework.http.converter.StringHttpMessageConverter">   
                <property name = "supportedMediaTypes">
                      <list>
                          <value>text/html;charset=UTF-8</value>   
                     </list>   
                </property>   
             </bean>   
         </list>   
   </property>  
</bean>   jquery

 

這樣經過配置AnnotationMethodHandlerAdaptermessageConverters屬性來指定編碼。
記住,須要把bean部分加入到<context:component-scan base-package="com.zlscw.mvc" />前面, web

這樣就能夠在jquery中直接調用而不出現亂碼了。 ajax

 

-------------------------------------------這篇文章說的很到位 spring

近日用Spring3的MVC寫東西,深感其之於Webwork/Struts2的便利,可是在經過@ResponseBody這個annotation輸出一個json字符串的時候,發現頁面上得到的json字符串中文字符出現了亂碼的現象。經過firefox觀察返回的字符串,中文部分所有變成了???????的形式,初步斷定是返回時,spring處理@ResponseBody使用了錯誤的編碼。 json

由於我在web.xml中已經配置了Spring的CharacterEncodingFilter,而且強制將request和response的編碼都指定爲utf-8,因此出現亂碼的緣由確定是在Spring內部某處的邏輯了。 瀏覽器

把log4j中關於spring的輸出級別調爲debug,經過訪問出問題的地址,發現Spring在處理@ResponseBody這個annotation的時候,org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter使用了org.springframework.http.converter.StringHttpMessageConverter進行處理,因而打開了Spring的源碼,看看這個類究竟作了哪些事情。 mvc

不看沒關係,一看嚇一跳,裏面居然是這樣定義其默認編碼的:  app

1 publicstaticfinalCharset DEFAULT_CHARSET = Charset.forName("ISO-8859-1");

頓時心生N種不爽:堂堂Spring,居然還在其中用西歐字符集做爲其默認編碼,坑爹啊!(不少spring的類中,涉及編碼的已經都是utf-8了,好比負責JSON視圖的MappingJacksonHttpMessageConverter,就是默認使用UTF-8)。原本想直接修改spring的源碼從新打包一個jar出來,後來看spring的java doc發現,其父類org.springframework.http.converter.AbstractHttpMessageConverter中的getDefaultContentType方法是能夠重寫的:

By default, this returns the first element of the supportedMediaTypes property, if any. Can be overridden in subclasses.

心想這下就簡單了,你的DEFAULT_CHARSET不是final麼?那我本身繼承一個出來,按照個人需求定義爲utf-8不就得了?代碼以下:

01 publicclassUTF8StringHttpMessageConverter extendsStringHttpMessageConverter {
02  
03  privatestaticfinalMediaType utf8 = newMediaType("text""plain",
04  Charset.forName("UTF-8"));
05  privatebooleanwriteAcceptCharset = true;
06  
07  @Override
08  protectedMediaType getDefaultContentType(String dumy) {
09  returnutf8;
10  }
11  
12  protectedList<Charset> getAcceptedCharsets() {
13  returnArrays.asList(utf8.getCharSet());
14  }
15  
16  protectedvoidwriteInternal(String s, HttpOutputMessage outputMessage)
17  throwsIOException {
18  if(this.writeAcceptCharset) {
19  outputMessage.getHeaders().setAcceptCharset(getAcceptedCharsets());
20  }
21  Charset charset = utf8.getCharSet();
22  FileCopyUtils.copy(s, newOutputStreamWriter(outputMessage.getBody(),
23  charset));
24  }
25  
26  publicbooleanisWriteAcceptCharset() {
27  returnwriteAcceptCharset;
28  }
29  
30  publicvoidsetWriteAcceptCharset(booleanwriteAcceptCharset) {
31  this.writeAcceptCharset = writeAcceptCharset;
32  }
33  
34 }

而後,在spring的配置文件中添加以下bean聲明,用本身寫的類替換掉原有的StringHttpMessageConverter:

1 <beanclass="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">
2     <propertyname="messageConverters">
3         <list>
4             <beanid="utf8StringHttpMessageConverter"class="xxx.xxx.UTF8StringHttpMessageConverter"/>
5         </list>
6     </property>
7 </bean>

再看經過@ResponseBody返回的json字符串,終於中文均可以正常顯示了。

 

-------------------------------------------下面有一些解釋

但咱們通常會在標註@ResponseBody的方法上返回String或byte[]類型的結果,指望的"Content-Type"的值應爲"text/plain"或"application/octet-stream"。
這樣致使了瀏覽器不能正確處理返回的內容。
實際上Spring在用HttpMessageConverter處理的過程當中首先會判斷response header中有沒有寫入"Content-Type",若是沒有寫入的話纔會使用request header中"Accept"的第一個值。
可是因爲Spring對HttpServletResponse進行了封裝,實際上使用的是ServletServerHttpResponse,這個類有一個對真正的HttpServletResponse的引用。
判斷response header的過程當中使用的是ServletServerHttpResponse的getHeaders()方法,但該方法並無返回真正的HttpServletResponse中的header。(這應該有問題吧?)
因此咱們雖然能夠在Controller的方法中加入對HttpServletResponse的引用,而後設置"Content-Type"的值,可是並不會起做用。
來處理@ResponseBody,該類再使用一些HttpMessageConverter來具體處理信息。
Chrome生成的值爲application/xml,application/xhtml+xml,text/html;q=0.9,text/plain;q=0.8,image/png,*/*;q=0.5
IE8生成的值爲application/x-ms-application, image/jpeg, application/xaml+xml, image/gif, image/pjpeg, application/x-ms-xbap, application/vnd.ms-excel, application/vnd.ms-powerpoint, application/msword, */*
因此最後寫入response中"Content-Type"的值爲"application/xml"或"application/x-ms-application"。

 

 -------------------------------------------------其實這個註解徹底能夠不用, 直接使用response往輸出流裏面寫。

 使用jQuery ajax調用的返回json,中文亂碼問題

Jquery :

$.ajax({
    url: '/test/testAction.do?method=test',
    type: 'POST',
    dataType: 'json',
    timeout: 5000,
    async: false,
    error: function(){
     alert('獲取數據失敗!');
    },
    success: function(json){
     jsObject  = eval(json);
    }
  });
  return jsObject;

 

JSONArray json = JSONArray.fromObject(SysList);//SysList是一個List //  設置response的ContentType解決中文亂碼   response.setContentType("text/html;charset=UTF-8");   response.getWriter().print(json.toString());   return null;

相關文章
相關標籤/搜索