因为我在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的源码,看看这个类究竟做了哪些事情。
不看不要紧,一看吓一跳,里面竟然是这样定义其默认编码的:
1 |
public static final Charset
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 |
public class UTF8StringHttpMessageConverter extends StringHttpMessageConverter
{ |
03 |
private static final MediaType
utf8 = new MediaType( "text" , "plain" , |
04 |
Charset.forName( "UTF-8" )); |
05 |
private boolean writeAcceptCharset
= true ; |
08 |
protected MediaType
getDefaultContentType(String dumy) { |
12 |
protected List<Charset>
getAcceptedCharsets() { |
13 |
return Arrays.asList(utf8.getCharSet()); |
16 |
protected void writeInternal(String
s, HttpOutputMessage outputMessage) |
18 |
if ( this .writeAcceptCharset)
{ |
19 |
outputMessage.getHeaders().setAcceptCharset(getAcceptedCharsets()); |
21 |
Charset
charset = utf8.getCharSet(); |
22 |
FileCopyUtils.copy(s, new OutputStreamWriter(outputMessage.getBody(), |
26 |
public boolean isWriteAcceptCharset()
{ |
27 |
return writeAcceptCharset; |
30 |
public void setWriteAcceptCharset( boolean writeAcceptCharset)
{ |
31 |
this .writeAcceptCharset
= writeAcceptCharset; |
然后,在spring的配置文件中添加如下bean声明,用自己写的类替换掉原有的StringHttpMessageConverter:
1 |
< bean class = "org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter" > |
2 |
< property name = "messageConverters" > |
4 |
< bean id = "utf8StringHttpMessageConverter" class = "xxx.xxx.UTF8StringHttpMessageConverter" /> |
再看通过@ResponseBody返回的json字符串,终于中文都可以正常显示了。