解决中文乱码的问题实例
解决中文乱码的问题实例
在进行web应用程序开发的过程中,中文问题经常困扰着很多程序员。因此,在接下来的两个课程的学习中,我将根据实践经验来谈一下web应用中的中文问题的解决方法。让大家不再为此而头痛。
我们就前那个登录验证的例子,进行进阶,处理中文乱码问题。前面例子中,只要用户名和密码都是123,才返回成功的页面。为了处理中文,我们判断只有用户名是张三才登陆成功。
一、程序进阶:
既然是中文乱码处理,页面的用户输入就是中文了,相应的action的if处理也要变成: if (username.equals("张三") && userpass.equals("123"));
效果图:
我们以结果为导向,首先大家先看看要完成的效果图,对接下来要做的事情有个清晰的认识。
1、如图键入:中文
显示结果如下图:
为什么会到错误页面呢?我们明明都写正确了呀,是的,没有错误,这是什么原因呢?
请看下图:
我们打印才发现原来username的值传到action中,成了乱码,这主要由于客户端和服务器端采用了不同的字符集,中文乱码我们没有处理。
二、解决办法:
A、直接转编码
我们新建一个包,命名为util,在包下新建一个类文件,命名为EncodingUtil,类的功能就是提供一个字符集转换的一个方法,具体代码如下所示:
package util;
public class Encoding {
public static String isToGB(String src) {
String strRet = null;
try {
strRet = new String(src.getBytes("ISO_8859_1"), "GBK");
} catch (Exception e) {
e.printStackTrace();
}
return strRet;
}
}
小结:这办法虽然能解决中文乱码,但是每次还得调用,是不是很不方便呢?如果忘记了调用这个方法,那程序又乱码了,维护起来很困难,下面我们看另一种解决方案。
B、继承RequestProcessor类
RequestProcessor类处理ActionServlet接收到的所有请求。根据它的处理方式,可将每个请求分解为多个小任务,分别由不同的方法执行。这就允许针对请求的各个单独部分自定义处理。
RequestProcessor类的部分方法如下:
processPath(): 获取客户端请求的路径URI
processMapping(): 根据请求URI获取所需的映射信息
processRoles(): 检查用户的角色是否允许他访问请求的资源
processActionForm(): 新建一个Form Bean或从请求会话中检索Form Bean
processForward(): 处理元素forward以匹配当前的请求路径
processValidate(): 调用Form Bean的validate()方法
processPreprocess(): 告诉请求处理器调用此方法后,是否应继续处理请求
processLocale(): 为请求选择一个语言环境
processActionCreate(): 实例化当前ActionMapping指定的类的实例
processActionPerform(): 将调用action的perform()或execute()方法
呵呵,发没发现RequestProcess类的所有方法都有一个前缀proess,接着往下看吧。
RequestProcessor在action之前,所以我们应着手RequestProcessor,要开发自己的RequestProcessor类,步骤如下:
(1) 创建一个继承org.apache.struts.action.RequestProcessor的类,在改类中显示定义一个无参,方法体为空的构造器。
(2) 重写所需的方法,加入我们的功能。
具体代码如下所示:
package servlets;
import java.io.UnsupportedEncodingException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.struts.action.RequestProcessor;
public class EncodingHandler extends RequestProcessor {
public boolean processPreprocess(HttpServletRequest servletRequest,
HttpServletResponse serveltResponse) {
try {
servletRequest.setCharacterEncoding("GBK");
System.out.println("请求被处理.");
} catch (UnsupportedEncodingException ex) {
ex.printStackTrace();
}
return true;
}
}
(3) 修改配置文件sturts-config.xml,在其中加入一个名为的元素,用以指定我们定制的RequestProcessor类。
path="/login"
name="loginActionForm"
scope="request"
type="actions.LoginAction">
上面就是我们自己的定义的RequestProcessor类,使用标签类定义。
如图:
小结:这样做呢?问题是解决了,每一个请求先经过这个方法,并转换了字符集再交给action做处理,这样我们不用在操心中文乱码,但RequestProcessor是与struts耦合在一块儿。如果不用struts框架,我们又该如何处理中文问题呢?是否又更好的办法呢?那就接着跟我往下看吧。
C、Filter来解决中文问题
Filter,是不是你脑子里闪现了这个词呢?下面就来看看如何用它来改写我们上一章节的例子吧!
(1) 首先在工程中新建一包,命名为filter,在下面新建一类文件,命名为EncodingServlet,并继承HttpServlet、实现Filter接口,注意并实现接口的方法。
在Servlet中filter起着过滤器的作用,当一个请求发送到服务器的时候,需要把请求首先交给filter来处理,然后交给action做处理。EncodingServlet负责处理请求的字符集,在此就起这么个功能,具体代码请依照如下所示:
package servlets;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServlet;
public class EncodingServlet extends HttpServlet implements Filter {
private static final long serialVersionUID = 1L;
public void doFilter(ServletRequest servletRequest,
ServletResponse serveltResponse, FilterChain filterChain) {
try {
servletRequest.setCharacterEncoding("GBK");
filterChain.doFilter(servletRequest, serveltResponse);
} catch (Exception ex) {
}
}
public void init(FilterConfig arg0) throws ServletException {
}
}
(2) 修改web.xml,加入我们的filter。
EncodingServlet
servlets.EncodingServlet
EncodingServlet
/*
小结:这个中文乱码处理用了fileter,而且适用与任何场合,比较实用。
在进行web应用程序开发的过程中,中文问题经常困扰着很多程序员。因此,在接下来的两个课程的学习中,我将根据实践经验来谈一下web应用中的中文问题的解决方法。让大家不再为此而头痛。
我们就前那个登录验证的例子,进行进阶,处理中文乱码问题。前面例子中,只要用户名和密码都是123,才返回成功的页面。为了处理中文,我们判断只有用户名是张三才登陆成功。
一、程序进阶:
既然是中文乱码处理,页面的用户输入就是中文了,相应的action的if处理也要变成: if (username.equals("张三") && userpass.equals("123"));
效果图:
我们以结果为导向,首先大家先看看要完成的效果图,对接下来要做的事情有个清晰的认识。
1、如图键入:中文
显示结果如下图:
为什么会到错误页面呢?我们明明都写正确了呀,是的,没有错误,这是什么原因呢?
请看下图:
我们打印才发现原来username的值传到action中,成了乱码,这主要由于客户端和服务器端采用了不同的字符集,中文乱码我们没有处理。
二、解决办法:
A、直接转编码
我们新建一个包,命名为util,在包下新建一个类文件,命名为EncodingUtil,类的功能就是提供一个字符集转换的一个方法,具体代码如下所示:
package util;
public class Encoding {
public static String isToGB(String src) {
String strRet = null;
try {
strRet = new String(src.getBytes("ISO_8859_1"), "GBK");
} catch (Exception e) {
e.printStackTrace();
}
return strRet;
}
}
小结:这办法虽然能解决中文乱码,但是每次还得调用,是不是很不方便呢?如果忘记了调用这个方法,那程序又乱码了,维护起来很困难,下面我们看另一种解决方案。
B、继承RequestProcessor类
RequestProcessor类处理ActionServlet接收到的所有请求。根据它的处理方式,可将每个请求分解为多个小任务,分别由不同的方法执行。这就允许针对请求的各个单独部分自定义处理。
RequestProcessor类的部分方法如下:
processPath(): 获取客户端请求的路径URI
processMapping(): 根据请求URI获取所需的映射信息
processRoles(): 检查用户的角色是否允许他访问请求的资源
processActionForm(): 新建一个Form Bean或从请求会话中检索Form Bean
processForward(): 处理
processValidate(): 调用Form Bean的validate()方法
processPreprocess(): 告诉请求处理器调用此方法后,是否应继续处理请求
processLocale(): 为请求选择一个语言环境
processActionCreate(): 实例化当前ActionMapping指定的类的实例
processActionPerform(): 将调用action的perform()或execute()方法
呵呵,发没发现RequestProcess类的所有方法都有一个前缀proess,接着往下看吧。
RequestProcessor在action之前,所以我们应着手RequestProcessor,要开发自己的RequestProcessor类,步骤如下:
(1) 创建一个继承org.apache.struts.action.RequestProcessor的类,在改类中显示定义一个无参,方法体为空的构造器。
(2) 重写所需的方法,加入我们的功能。
具体代码如下所示:
package servlets;
import java.io.UnsupportedEncodingException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.struts.action.RequestProcessor;
public class EncodingHandler extends RequestProcessor {
public boolean processPreprocess(HttpServletRequest servletRequest,
HttpServletResponse serveltResponse) {
try {
servletRequest.setCharacterEncoding("GBK");
System.out.println("请求被处理.");
} catch (UnsupportedEncodingException ex) {
ex.printStackTrace();
}
return true;
}
}
(3) 修改配置文件sturts-config.xml,在其中加入一个名为
path="/login"
name="loginActionForm"
scope="request"
type="actions.LoginAction">
上面就是我们自己的定义的RequestProcessor类,使用
如图:
小结:这样做呢?问题是解决了,每一个请求先经过这个方法,并转换了字符集再交给action做处理,这样我们不用在操心中文乱码,但RequestProcessor是与struts耦合在一块儿。如果不用struts框架,我们又该如何处理中文问题呢?是否又更好的办法呢?那就接着跟我往下看吧。
C、Filter来解决中文问题
Filter,是不是你脑子里闪现了这个词呢?下面就来看看如何用它来改写我们上一章节的例子吧!
(1) 首先在工程中新建一包,命名为filter,在下面新建一类文件,命名为EncodingServlet,并继承HttpServlet、实现Filter接口,注意并实现接口的方法。
在Servlet中filter起着过滤器的作用,当一个请求发送到服务器的时候,需要把请求首先交给filter来处理,然后交给action做处理。EncodingServlet负责处理请求的字符集,在此就起这么个功能,具体代码请依照如下所示:
package servlets;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServlet;
public class EncodingServlet extends HttpServlet implements Filter {
private static final long serialVersionUID = 1L;
public void doFilter(ServletRequest servletRequest,
ServletResponse serveltResponse, FilterChain filterChain) {
try {
servletRequest.setCharacterEncoding("GBK");
filterChain.doFilter(servletRequest, serveltResponse);
} catch (Exception ex) {
}
}
public void init(FilterConfig arg0) throws ServletException {
}
}
(2) 修改web.xml,加入我们的filter。
小结:这个中文乱码处理用了fileter,而且适用与任何场合,比较实用。