在学习javaweb过程中由于软件设计者都是外国人,他们所支持的编码都是ISO8859-1对于我们中国人来讲必须是中文,所以我们一般使用的编码就是utf-8或者GBK,然而开发工具都是用ISO8859-1编码,我们用utf-8解码的话就会产生乱码。
解决响应乱码一般都很好解决只需要加下面的一段代码就可以:
response.setContentType("text/html;charset=utf-8");
然而对于请求乱码的解决根据请求方式的不同解决乱码的方式也不同,通常我们常用的是POST和GET这两种方式。对于POST方式使用下面的代码就可以解决:
request.setCharacterEncoding("utf-8");
但是对于get方式的请求方式上面的代码就不能解决了,get的解决方案是将iso8859-1的编码通过反编码,再通过utf-8解码就可以解决:
str=new String(str.getByte("iso8859-1"),"utf-8");
所以在servlet的doGet方法中就可以解决乱码问题(POST提交方式):
<span style="white-space:pre"> </span>public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
request.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=utf-8");
}
但是在这里就存在一个问题,每创建servlet都需要加上这样一段代码,还需要知道是POST提交还是GET提交,这样就显得很繁琐,需要解决这个问题就要用到web开发中另一个重要的知识点-过滤器。
Filter是Servlet技术中最实用的技术,WEB开发人员通过Filter技术,对web服务器管理的
所有
web
资源:例如
Jsp
, Servlet,
静态图片文件或静态
html
文件等进行
拦截,从而实现一些特殊的功能。例如实现URL级别的权限访问控制、过滤敏感词汇、压缩响应信息等一些高级功能。
开发一个Filter分两个步骤:
一、写一个类实现特定的接口Filter;
二、在web.xml文件中注册这个Filter。
在实现的类中处理乱码,当所有的资源访问时都会通过这个过滤器过滤,从而达到解决乱码的目的。
<span style="white-space:pre"> </span><filter>
<description>全局编码过滤器</description>
<filter-name>EncodingFilter</filter-name>
<filter-class>com.ruijie.filter.EncodingFilter</filter-class>
<init-param>
<param-name>encode</param-name>
<param-value>utf-8</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>EncodingFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<span style="white-space:pre"> </span>public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
response.setContentType("text/html;charset="+encode); //解决响应乱码
chain.doFilter(new MyHttpServletRequest((HttpServletRequest) request), response);
}
上文也说了在解决乱码过程中get方式提交的方式是需要主要解决的,在获取参数过程中通常是通过以下三个方法获取:
<span style="white-space:pre"> </span>getParameterMap()
getParameterValues(String name)
getParameter(String name)
所以问题就转移到解决这三个方法的乱码问题就可以解决get方式的乱码问题。解决方案是将ServletRequest进行改造成MyHttpServletRequest,可以通过两种方式进行改造一种是装饰模式,一种是动态代理;由于为我们提供了HttpServletRequestWrapper,所以在这里我们使用装饰者模式更加方便
<span style="white-space:pre"> </span>class MyHttpServletRequest extends HttpServletRequestWrapper{
private HttpServletRequest request = null;
private boolean isNotEncode = true;
public MyHttpServletRequest(HttpServletRequest request) {
super(request);
this.request = request;
}
@Override
public Map getParameterMap() {
try{
if(request.getMethod().equalsIgnoreCase("POST")){//--如果是post提交
request.setCharacterEncoding(encode);
return request.getParameterMap();
}else if(request.getMethod().equalsIgnoreCase("GET")){//--如果是get提交
Map<String,String[]> map = request.getParameterMap();
if(isNotEncode){//只能在第一次解决乱码
for(Map.Entry<String, String[]> entry : map.entrySet()){
String [] vs = entry.getValue();
for(int i=0;i<vs.length;i++){
vs[i] = new String(vs[i].getBytes("iso8859-1"),encode);
}
}
isNotEncode = false;//设置为false,第二次就不会再进这个代码块了
}
return map;
}else{
return request.getParameterMap();
}
}catch (Exception e) {
e.printStackTrace();
throw new RuntimeException(e);
}
}
@Override
public String[] getParameterValues(String name) {
return (String[]) getParameterMap().get(name);
}
@Override
public String getParameter(String name) {
return getParameterValues(name) == null ? null : getParameterValues(name)[0];
}
}