springmvc ajax中文乱码,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:

text/html;charset=UTF-8

jquery

这样经过配置AnnotationMethodHandlerAdapter类messageConverters属性来指定编码。

记住,须要把bean部分加入到前面, 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 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

2

3

4

5

6

7

再看经过@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,**

因此最后写入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_r(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;

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值