先分析一下出现乱码的几种情况:
- 数据在页面显示有乱码
- 数据到服务器后有乱码
- 从服务器返回的有乱码
- 使用Ajax收发时有乱码
数据在页面显示有乱码
也就是服务器获取数据时没有问题,在客户端上显示数据为乱码,这种情况最好解决,只需修改页面的编码即可:
pageEncoding="utf-8"
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
数据到服务器后有乱码
通常出现在表单提交数据给服务器,服务器获取时得到的是乱码,以Tomcat为例,出现这种情况是因为没有告诉服务器以什么编码去获取数据,假设页面是以utf-8的编码显示的数据,你填写时没有问题,看起来是没有问题,但是计算机只认识0和1,它在传输过程中是以二进制的样式进入到服务器的,这时你的服务器就不知道它之前是什么编码,如果你没有指定,它就会按照老外的喜好,以iso8859-1的编码去读取,结果就出乱码了.
这种情况的解决方式又要一分为二,为POST和GET方式,POST解决方案很简单,既然获取时有乱码,那么在获取之前,设置编码即可:
request.setCharacterEncoding("utf-8");
然后再使用传统方法获取:
String value = request.getParameter("value");
需要注意,这种设置只对POST提交有效,如果是GET则相对麻烦一点.因为乱码是在调用方法getParameter()时出现的,方法内部的编码肯定出现了问题,既然它不能自动使用UTF-8,那我们就手动来转:
if(request.getMethod().equalsIgnoreCase("get")){
value = new String(value.getBytes("iso8859-1"),"utf-8");
}
由于POST提交方式已经得到了解决,我们就只需要在提交方式为GET时进行处理.这样,就能保证服务器获取的数据肯定不是乱码
从服务器返回的有乱码
通常只需要设置response的编码:
response.setCharacterEncoding("utf-8");
然后再设置浏览器的编码为UTF-8
或者通知浏览器以UTF-8的编码打开
response.setContentType("text/html;charset=utf-8");
这样就能保证浏览器收到的是UTF-8的数据,并且显示也没有问题.
使用Ajax收发时有乱码
如果是使用Ajax或者JavaScript提交的表单,是用get就需要对传递的中文进行编码.使用JS的内置函数encodeURI(),这个函数把URI字符串采用UTF-8编码格式转化成escape格式的字符串,如果不使用该函数,将由浏览器进行默认的编码,但这是一种不能预测的行为.总而言之,只要上述问题解决了,Ajax也就清静了
另外,如果获取数据时,服务器端的每个程序都需要针对POST,GET进行设置,且都是同样的写法,那就显得太不优雅了,所以最后用一个过滤器替换掉对所有页面的request和response的设置.那我就需要在过滤器里面对request或者response对象进行增强,然后把增强后的request或者response放行,我在服务器程序里面就不用再写解决乱码的代码,从而提高程序的可维护性.
增强一个类有三种方式:
- 直接继承该类
- 包装设计模式
- 动态代理
显然,这里不适合用继承,虽然它最简单,但是一个request对象中包含着众多的数据,如果想创造一个request对象,就必须知道request是如何产生的,然后用服务器创造request对象的方式去创造我的"MyRequest"对象,来达到增强的目的.所以在这里放弃这种方式.
过滤器+包装类
public class CharacterFilter implements Filter {
public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {
HttpServletRequest request = (HttpServletRequest) req;
HttpServletResponse response = (HttpServletResponse) res;
//POST的乱码解决方案
request.setCharacterEncoding("utf-8");
//返回数据的乱码解决方案
response.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=utf-8");
//将增强后的对象放行
MyRequest myRequest = new MyRequest(request);
chain.doFilter(myRequest, response); //这样一来,后面所有的操作都是基于这个增强后的对象进行的
}
/**创建一个request对象的包装类:
1.编写一个类,实现与被增强对象相同的接口
2.在类中定义一个变量,记住被增强对象
3.在类中定义一个构造方法,接收被增强对象
4.覆写想要增强的方法
5.对于不想增强的方法,直接调用被增强对象(目标对象)的方法
包装设计模式"五步曲"
*/
class MyRequest extends HttpServletRequestWrapper{
private HttpServletRequest request;
public MyRequest(HttpServletRequest request) {
super(request);
this.request = request;
}
@Override
public String getParameter(String name) {
//如果请求方式是POST,则不用增强,直接调用目标对象的方法
if(this.request.getMethod().equalsIgnoreCase("POST")){
return this.request.getParameter(name);
}
/*程序运行到此,请求方式必然为GET
先获取值,再进行手动转换*/
String value = this.request.getParameter(name);
try {
value = new String(value.getBytes("iso8859-1"),"utf-8");
} catch (UnsupportedEncodingException e) {
throw new RuntimeException(e);
}
return value; //返回
}
}
public void destroy() {
// TODO Auto-generated method stub
}
public void init(FilterConfig fConfig) throws ServletException {
// TODO Auto-generated method stub
}
}
过滤器记得在web.xml中配置一把
过滤器+动态代理
public class CharacterFilter2 implements Filter {
public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {
final HttpServletRequest request = (HttpServletRequest) req;
HttpServletResponse response = (HttpServletResponse) res;
//POST的乱码解决方案
request.setCharacterEncoding("utf-8");
//返回数据的乱码解决方案
response.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=utf-8");
//用动态代理拦截,增强getParameter()后,放行
chain.doFilter((ServletRequest) Proxy.newProxyInstance(CharacterFilter2.class.getClassLoader(), request.getClass().getInterfaces(),
new InvocationHandler(){ //直接实现接口
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
//如果请求方式是POST,则不用增强,直接调用目标对象的方法
if(request.getMethod().equalsIgnoreCase("POST")){
return method.invoke(request, args);
}
String methodName = method.getName();
//如果传递进来的方法不是getParameter(),则不用增强
if(!methodName.equals("getParameter")){
return method.invoke(request, args);
}
//为GET,并且是getParameter(),...
String value = (String) method.invoke(request, args);
if(value!=null){
value = new String(value.getBytes("iso8859-1"),"utf-8");
}
return value;
}
}), response);
}
public void destroy() {
// TODO Auto-generated method stub
}
public void init(FilterConfig fConfig) throws ServletException {
// TODO Auto-generated method stub
}
}
好了,还是记得要配置一把
至此,全站的乱码问题得到解决,其实还有一个更为简单的方式,就是修改服务器的配置文件,但是如果服务器不受自己控制,就没用了.