JAVA乱码


======================================================
在最后,我邀请大家参加新浪APP,就是新浪免费送大家的一个空间,支持PHP+MySql,免费二级域名,免费域名绑定 这个是我邀请的地址,您通过这个链接注册即为我的好友,并获赠云豆500个,价值5元哦!短网址是http://t.cn/SXOiLh我创建的小站每天访客已经达到2000+了,每天挂广告赚50+元哦,呵呵,饭钱不愁了,\(^o^)/
======================================================

开发java应用出现乱码是很常见的,毕竟现在unicode的使用还不是很广泛,在使用gb2312(包含了gbk简体,big5繁体)的系统中要正确实现

中文的display和数据库的存储是最基本的要求。

==============================

1,首先developer要明确自己为什么会遇到乱码,遇到什么样的乱码(无意义的符号还是一串问号或者其它什么东西)。

新手遇到一堆很乱的字符时通常不知所措,最直接的反映就是打开google搜索”java中文”(这个字符串在搜索引擎上的查询频率非常高),

然后一个一个的去看别人的解决方法。这样做没有错,但是很难达到目的,原因下面会提到。

总之,出现乱码的原因是非常多的,解决的方法也完全不一样,要解决问题必须先分析自己的”上下文环境”。

============================

2,具体说来,需要哪些信息才能确定项目中的乱码的根源。

a,开发者所用的操作系统

b,j2ee容器的名称,版本

c,数据库的名称,版本(精确版本)以及jdbc驱动的版本

d,出现乱码的source code(比如是system out 出来的,还是jsp页面中的,如果是jsp中的,那么头部声明的情况也很重要)

===========================================================

3,如何初步分析乱码出现的原因。

有了上述的信息,基本上就可以发帖求助了,相信放到javaworld等论坛上,很快就会有高手给你提出有效的解决方案的。

当然不能总靠发帖求助,也要试试自行解决问题。如何下手呢?

a,分析一下你的”乱码”到底是什么编码。这个其实不难,比如

system.out.println(teststring);

这一段出现了乱码,那么不妨用穷举法猜测一下它的实际编码格式。

system.out.println(new string(teststring.getbytes(”iso-8859-1″),”gb2312″));

system.out.println(new string(teststring.getbytes(”utf8″),”gb2312″));

system.out.println(new string(teststring.getbytes(”gb2312″),”gb2312″));

system.out.println(new string(teststring.getbytes(”gbk”),”gb2312″));

system.out.println(new string(teststring.getbytes(”big5″),”gb2312″));

等等,上述代码的意思是用制定的编码格式去读取teststring这个”乱码”,并转换成gb2312(此处仅以中文为例)

然后你看哪一个转换出来的结果是ok的,那就。。。

b,如果用上面的步骤能得到正确的中文,说明你的数据肯定是在的,只不过是界面中没有正确显示而已。那么第二步就该纠正你的view部分了

,通常需要检查的是jsp中是否选择了正确的页面编码。

在此要声明被很多人误解的一点,那就是指令和两者的不同。通常网上的很多文章在提到中文问题时都是说数据库中选择unicode或者gb2312存储,同

时在jsp中用page指令声明编码就可以解决。但是我觉得这种说法很不负责任,害的我费了n多时间为本来并不存在的乱码而郁闷。实际上page

的作用是在jsp被编译成为html的过程中提供编码方式让java来”读取”表达式当中的string(有点类似于上面的第三个语句的作用),而meta

的作用是众所周知的为ie浏览器提供编码选择,是用来”显示”最后的数据的。但是没有看到有人提醒这一点,我一直把page当成meta在用,

导致本来是iso-8859的数据,被page指令读成gb2312,于是乱码,所以又加了编码转化的函数把所有的string数据都从iso8859转到gb2312(为

什么这么转,当时也没考虑这么多,因为这么做可以正常显示了,所以就这么改了,呵呵当时实在没有时间慢慢排查问题了)。

4,数据库选择什么样的编码比较好。

目前流行的db主要有sql server,mysql,oracle,db2等,其中mysql作为免费db中的老大,性能和功能是得到公认的,安装配置比较方便,相

应的driver也比较完善,性价比是绝对的ok。所以就以mysql为例。

我个人建议采用mysql的默认编码来存储,也就是iso-8859-1(在mysql的选项中对应于latin-1)。理由主要有这么几个,一是iso-8859-1对中

文的支持不错;二是跟java中的默认编码一致,至少在很多地方免除了转换编码的麻烦;三是默认的比较稳定,兼容性也更好,因为多编码的

支持是由具体的db产品提供的,别说跟其它的db会不兼容,即使自身的不同版本也可能出现兼容性的问题。

例如mysql 4.0以前的产品中,很多中文的解决方案是利用connection中的characterencoding字段来制定编码,比如gb2312什么的,这样是ok

的,因为原数据都是iso8859_1编码,jdbc驱动会采用url里面指定的character set来进行编码,resultset.getstring(*)取出的就是编码后的

字符串。这样就直接拿到gb2312的数据了。

但是mysql 4.1的推出给很多dbadmin带来了不小的麻烦,因为mysql4.1支持column level的character set,每个table,column都可以指定编码

,不指定就是iso8895_1,因此jdbc取出数据后会根据column的character set来进行编码,而不再是用一个全局的参数来取所有的数据了。

这从另一个方面也说明了乱码问题的产生实在是很复杂的事情,原因太多了。我也只是针对自己遇到的实际情况提供一些解决思路。

我一个朋友要我帮它改改,然后就采用了上面的方法:解决方法如下

http://www.family168.com/tutorial/jsp/html/jsp-ch-02.html

string name=request.getparameter("name").tostring();

system.out.println("转换前的姓名:"+name);

//采用枚举法测试

//system.out.println("111111"+new string(name.getbytes("iso-8859-1"),"gb2312"));

//system.out.println(new string(name.getbytes("utf-8"),"gb2312"));

//system.out.println(new string(name.getbytes("gb2312"),"gb2312"));

//system.out.println(new string(name.getbytes("gbk"),"gb2312"));

//system.out.println(new string(name.getbytes("big5"),"gb2312"));

//此处必须转换成中文字符集,这个地方和数据库字符集是有关系的,是一一对应的。

byte[] bytes = name.getbytes("iso-8859-1");

string result = new string(bytes, "gb2312");

system.out.println("转换后的姓名:"+result);

以上就是解决方法。很有效。


======================================================
在最后,我邀请大家参加新浪APP,就是新浪免费送大家的一个空间,支持PHP+MySql,免费二级域名,免费域名绑定 这个是我邀请的地址,您通过这个链接注册即为我的好友,并获赠云豆500个,价值5元哦!短网址是http://t.cn/SXOiLh我创建的小站每天访客已经达到2000+了,每天挂广告赚50+元哦,呵呵,饭钱不愁了,\(^o^)/
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值