【问题】:
使用spingmvc,在JS里面通过ajax发送请求,并返回json格式的数据;
从数据库拿出来是正确的中文格式,展示在页面上就是乱码。
乱码格式都是???
后台断点调试:中文格式正常显示。
前端控制台显示:
【原因】:
spring mvc的一个bug,spring MVC有一系列HttpMessageConverter去处理用@ResponseBody注解的返回
值。如:返回list或其它,使用 MappingJacksonHttpMessageConverter;返回string,则使用
StringHttpMessageConverter,而这个convert使用的是字符集是iso-8859-1,而且是final的。所以在当返回json
中有中文时会出现乱码。
【解决】:
尝试方法一:
在@RequestMapping里的并发配置produces={"text/html;charset=UTF-8;"}
@RequestMapping(value = "***",produces="text/html;charset=UTF-8;")
(value = "***",produces="text/html;charset=UTF-8;")
用Ajax访问,回的是json格式,所以没有起作用还是乱码。
[原理]手动给对应的Accept返回制定格式编码数据。
尝试方法二:
在@RequestMapping里的并发配置produces={"application/json;text/html;charset=UTF-8;"}
@RequestMapping(value = "***",produces={"application/json;text/html;charset=UTF-8;"})
网上解释该方式都可行,但是我还不可行。
[原理]手动给对应的Accept返回制定格式编码数据。
尝试方法三:
暴力解决:
直接将org.springframework.http.converter.StringHttpMessageConverter 里面的属性defaultCharset设置成utf-8
<mvc:annotation-driven >
<!-- 消息转换器 -->
<!-- 直接将org.springframework.http.converter.StringHttpMessageConverter
里面的属性defaultCharset设置成utf-8 -->
<mvc:message-converters register-defaults="true">
<bean class="org.springframework.http.converter.StringHttpMessageConverter">
<constructor-arg value="UTF-8"/>
</bean>
</mvc:message-converters>
</mvc:annotation-driven>
当然百度看到还有一种更加 更加 暴力解决方式:(没有用)
方法四:拿到源代码,修改成UTF-8后再重新打包到spring-web-3.2.2.jar
public static final Charset DEFAULT_CHARSET = Charset.forName("ISO-8859-1");
改为
public static final Charset DEFAULT_CHARSET = Charset.forName("UTF-8");
方式三 正确解决!!!
【感悟】:
该乱码问题,定位了蛮久,最开始编码都是千篇一律得改文件编码格式,tomcat 编码,控制台的输出编码,也检查了JSP的编码是否设置,后面定位到Ajax 返回 json出现 的编码问题,给Ajax加
contentType: "application/x-www-form-urlencoded; charset=utf-8",
也不行,在方法内部 强制手动转编码
response.setContentType("application/json;charset=utf-8");
response.setcharEncoding("utf-8");
也不行。
断点调试,发现会调用 StringHttpMessageConverter 。百度一波,找到真正的 原因后,发现是 注解@ResponseBody返回值 springMVC 的StringHttpMessageConverter 的默认编码,引起的。