关注技术,关注生活

任何事情只要开始去做,永远不会太迟。
posts - 5, comments - 23, trackbacks - 0, articles - 18
  BlogJava :: 首页 :: 新随笔 :: 联系 :: 聚合  :: 管理

[原创]struts,ajax乱码解决方案

Posted on 2006-12-09 21:24 errorfun 阅读(3844) 评论(12)  编辑  收藏 所属分类: JavaAjax

乱码问题好像跟我们中国程序员特别有缘,一直困扰着我们,从开始的JSP乱码问题,STRUTS乱码问题,到现在的AJAX乱码问题,无一不是搞得许多程序员焦头烂额的,整天骂XXX产品对中文支持不了,UTF-8无法使用中文啊什么的,其实这里面被骂的产品中其实99%以上是对中文支持非常好的,而出现乱码的原因只是因为自身对国际化支持以及文件编码等信息的认识不知造成的。要知道一个产品那么流行,怎么可能对中文支持不了呢,下面就开始一一帮大家解决这些问题。

1
、编码
      --
想要解决好中文问题,对编码肯定是不能一概不懂了,编码是解决中文乱码问题的根本。
      
编码比较常用的有: UTF-8 GBK GB2312 ISO-8859-1 ,除了 iso-8859-1 之外的其它三个编码都能很好的支持中文,但它们都兼容 ISO-8859-1 的编码(就是说无论编码怎么改变,只要是 ISO-8859-1 中的字符,永远不会出现乱码)。
     
这四种编码中, GB2312 是中国规定的汉字编码,也可以说是简体中文的字符集编码 ; GBK GB2312 的扩展 , 除了兼容 GB2312 外,它还能显示繁体中文,还有日文的假名 ; UTF-8 虽然也支持中文,但却 GB 码不兼容(编码值不同) UTF-8 使用的是可变长的 UNICODE 编码,编码可能是 1 16 进制(即 ISO-8859-1 中的字符,其编码也是相同的)也有可能是 2 位或 3 位的 16 进制。 UTF-8 的优点是: 1 CPU 字节顺序无关 , 可以在不同平台之间交流。 2 、容错能力高 , 任何一个字节损坏后 , 最多只会导致一个编码码位损失 , 不会链锁错误 ( GB 码错一个字节就会整行乱码 ) ,所以在国际化处理中基本都是建议使用 UTF-8 作为编码。

2、文件的编码
      --虽然说只要设置了正确的编码就可以使字符正确显示了,但如果忽略了文件保存时的编码的话,那可是会让你走进迷雾中的。
      文件编码最常使用的有两种:ANSI和UTF-8,光看名字估计你都可以猜到了,ANSI就是我们保存文件时使用的默认编码,而UTF-8则需自己设置。对于编码的改变,我使用的工具是NOTEPAD和ECLIPSE,NOTEPAD使用最简单,只要打开文件后在另存为中选择相应的编码就行了,而且它对编码的支持非常好;而在ECLIPSE中,只要稍微设置一下就行了,打开首选项,然后选择:常规->内容类型(ContentType),在右边选中你想改变保存编码的文件类型,然后在下方的缺省编码中改变其值,最后点击更新(UPDATE)按钮即可。



而在其它的编辑器中,默认保存的内容都是GB2312或者GBK(NOTEPAD中对应ANSI).而根据前面所说的UTF-8和GBK,GB2312等的编码值是不同的这一点,可以知道,如果文件使用了UTF-8,那么字符编码就必须使用UTF-8,否则编码值的不同就可能造成乱码。而这也就是为什么那么多的人使用了UTF-8编码后还会产生乱码的根本原因。(JS和JSP都是这个道理)

3、JSP,STRUTS等的中文乱码解决方案
     其实解决的方法只有一个:

 request.setCharacterEncoding(encoding);

   方法只有一种,但处理方式就多种多样了,初学者会在JSP页面上直接使用,而有经验的程序员会使用过滤器。而现在所要说的方法也是过滤器。这里以统一使用UTF-8作为编码作为例子说明。具体过程就不多说了,网上有很多教程。偷懒一点的,到TOMCAT中复制就行了。在TOMCAT的目录下的\webapps\jsp-examples\WEB-INF\classes\filters\找到SetCharacterEncodingFilter.java 这个类,放到你的程序中并配置好映射路径。配置好后基本上你的乱码问题就解决了。但要映射路径中需要注意的就是不能使用 '*'

   < filter-mapping >
    
< filter-name > Set Character Encoding </ filter-name >
    
< servlet-name > * </ servlet-name >
  
</ filter-mapping >

像上面这样配置的话(可能也是网上大多教程的做法,想当年也是害苦了我),可能你只有JSP的乱码解决了,要解决STRUTS的乱码需要映射 *.do 或者 servletActionName。然后在初始化参数中设置encoding的值就行了。

< init-param >
      
< param-name > encoding </ param-name >
      
< param-value > UTF-8 </ param-value >
</ init-param >

当然,最重要的是要记得根据前面所说的方法,改变你所使用的编辑器保存文件的编码要与使用的字符编码一致。
而在JSP内容中,还是使用如网上教程所说的那种技俩,在所有页面的页首加入:

<% @ page language = "java" contentType = " text / html; charset = UTF - 8 "
    pageEncoding
= "UTF - 8 " %>

至此,相信JSP,ACTION都不太可能出现乱码了。

4、资源文件的乱码解决方案
      资源文件谁都知道是国际化支持不可或缺的一部分,如果资源文件都出现乱码了那还了得?其实资源文件的乱码是很好解决的,其原因也是因为使用了UTF-8做为JSP编码后,没有相应地改变资源文件的文件编码造成的,所以只要对资源文件保存的编码进行更正后,乱码问题也就解决了。当然,你的中文要使用 native2ascii 命令进行正确的转换。

5、调用JS时,JS内容乱码的解决方案。
     其实JS的乱码还是跟文件的编码有关系的,如果JS中有中文的话,那JS文件保存的编码就必须跟调用此JS的页面编码相同,否则,你的所有中文都要从JSP页面传给JS才会显示正常。可以看出对于调用JS出现的乱码是最容易解决的(也是建立在前面的辛苦之下的)。

6、AJAX提交数据乱码,返回数据乱码的解决方案
     随着AJAX的流行,乱码问题也开始困扰着许多刚开始使用它的程序员,幸好我之前对JSP乱码有过一点研究,在遇到AJAX后,并没有给我带来多大的困扰,在此将我的一些心得共享给大家。
     万变不离其宗,AJAX的乱码问题自然跟编码有关了,其实很多人跟我一样想到了对文件编码进行设置,并且在接数据时设置了requet的编码,在返回的数据时设置了response的编码一切都以为会很顺利,可是这一切都是徒劳无功的,讨厌的乱码再一次出现在你眼前。在你试了N多种方法,包括JS自身的escape,unescape方法后,你发现乱码仍然猖狂地出现在屏幕上。
    其实在试过这N多方法后,很多人都没发现,解决的方法其实很简单,而且其答案就在我们之前处理的JSP乱码之中。让我们先看一下AJAX的经典请求代码

xmlhttp.open( "post", url, async );
xmlhttp.setRequestHeader( 
"Content-Type""text/html"
 );
xmlhttp.send( params );

通过前面的说明,不知道你现在看出端倪了没有。不知道是受了网上教程的影响还是其它方面影响,setRequestHeader并是万年不变的,也没人想过去改它,而问题就正好出在这个地方。回想一个JSP页面内容的编码设置,其中有这么一节:
contentType="text/html; charset=UTF-8"

现在知道问题了吧,所以我们要把第二句代码改为:
xmlhttp.setRequestHeader( "Content-Type""text/html;charset=UTF-8" );

最后别忘了在返回数据时也设置上:
response.setContentType( "text/xml" );
response.setCharacterEncoding( 
"UTF-8" );

是不是很简单,一点都不麻烦呢?
如果要问为什么的话,其实我们可以把xmlhttp看成是一个临时页面,它由浏览器动态生成,主要作用是在后台获得请求的数据(可以看成是一个高级的iframe)。所以对于普通页面设置的编码,对它也要同样设置。而在servlet中返回数据为什么要设置contentType和encoding其道理也是一样的。众所周知,jsp的最后形态就是servlet,而jsp页首设置的那个内容其实也就是让生成的servlet中生成这么两句话:
response.setContentType( "text/html" );
response.setCharacterEncoding( 
"UTF-8" );

而pageEncoding则是跟jvm说明了这个页面的内容要使用什么编码保存(这跟之后生成的CLASS有关系)。所以在servlet设置response的编码也是理所当然的了。

一口气把自己一年以来遇到的乱码问题和解决的方案写出来了,希望对你有所帮助。

评论

# re: [原创]struts,ajax乱码解决方案  回复  更多评论   

2006-12-10 02:40 by jackstudio
小弟将楼主的文章转到了自己的Blog了,
希望楼主支持。

# re: [原创]struts,ajax乱码解决方案  回复  更多评论   

2006-12-10 12:33 by errorfun
支持

# re: [原创]struts,ajax乱码解决方案  回复  更多评论   

2006-12-11 11:02 by Zou Ang[匿名]
很好的文章,转到我的Blog上做珍藏,楼主如果不同意,麻烦告知我,马上删除

# re: [原创]struts,ajax乱码解决方案  回复  更多评论   

2006-12-11 12:12 by errorfun
没关系的,写出来的东西就是要让大家共同学习的,如果不让转载,那就没意义了

# re: [原创]struts,ajax乱码解决方案  回复  更多评论   

2006-12-11 19:26 by BeanSoft
楼主真是个好人!!!

# re: [原创]struts,ajax乱码解决方案  回复  更多评论   

2006-12-21 16:42 by javaMan[匿名]
楼主真是个不错的人!!
大力支持!!!!!谢谢你无私的精神!!

# re: [原创]struts,ajax乱码解决方案  回复  更多评论   

2006-12-24 15:53 by BeanSoft
ajax 的客户端提交表单数据的时候用 encodeURIComponent 比较保险. 还有就是建议统一使用 UTF-8 编码. 近期将发一份关于前一阵子做应用的一个例子.

# re: [原创]struts,ajax乱码解决方案  回复  更多评论   

2007-07-06 12:20 by CAnca
俺转到自己博客啦!!如果楼主不同意,俺立即删除啦!

# re: [原创]struts,ajax乱码解决方案  回复  更多评论   

2008-10-12 10:54 by sitinspring
有些乱码没那么简单,比如url中嵌入中文再取出最后一个字会变成问号。

# re: [原创]struts,ajax乱码解决方案  回复  更多评论   

2008-10-15 00:05 by errorfun
@sitinspring

你说的这个还是编码的问题,中文取出后变成问号就是和我说的第6点一样的问题,一般情况下有可能出现的就是你的URL中文用的是UTF-8但提交时可能把它当成GBK了,或者是GBK当成UTF-8了,这时候会有部分不出出现错误,但有一些会出现?或方框,这是因为UTF-8中的码表跟GBK的不是一样的,但有部分一样。而出现?大多数情况下是转成ISO-8859-1出问题,出方框是转成GBK出问题,这部分因为要说起来会很麻烦,所以我也没在这里面提出来,但只要你在所有地方设置好了编码,一般就不会出现这种情况了。

还有你这种情况的出现,有时是你在TOMCAT里没设置好编码造成的,这个配置一下就行了的。

# re: [原创]struts,ajax乱码解决方案  回复  更多评论   

2008-10-15 19:23 by sheng
楼主ajax部分的乱码解决方法有没有测试过的?
发现有以下几个问题:
1.很多书上说到用post提交时必须是大写的"POST";
2.与POST相对应的contentType必须是"application/x-www-form-urlencoded",否则提交不了数据,我做过测试的确是这样;
3.只在服务端和客户端设置相应header还是解决不了乱码问题,还是得显式进行转换。如下:
String para = request.getParameter("paraName");
byte[] tmp = para.getBytes("ISO8859-1");
para = new String(tmp,"GBK");
//如果是form表单直接提交的数据就不须这样转换也不会乱码,
//但如果是用url直接提交的或以ajax提交的参数都必须进行以上
//显式转换。 不知楼主有没有试过这种情况?

# re: [原创]struts,ajax乱码解决方案  回复  更多评论   

2008-10-15 19:40 by sheng
我正在做一个项目,用到struts+ajax.
用ajax提交的表单数据到ActionForm中,中文的都变成乱码。
虽然可以用显式的一个参数一个参数进行转换,但那已失去了ActionFrom自动
封装的意义。
请楼主帮忙想想办法!!!

说明:
request.setCharacterEncoding("GBK");
contentType="text/html; charset=gb2312"
及SetCharacterEncodingFilter.java
等文件头设置的东东都已用上了。而且用表单提交不会出乱码也不用显式转码。只有ajax中才会乱码。ajax中已设置了文件头,如下:
POST:requestObj.setRequestHeader("Content-Type", "application/x-www-form-urlencoded;charset=GBK");

GET:requestObj.setRequestHeader("Content-Type", "text/html;charset=GBK");

用GET方法提交时需显式转换字符集,用POST提交时显式转换字符集也没用。

只有注册用户登录后才能发表评论。


网站导航: