超强过滤器彻底解决JSP/SERVLET中文参数GET/POST传递的问题

来源:互联网 发布:magic launcher 知乎 编辑:程序博客网 时间:2024/04/30 04:07
 

一:首先学到的解决方法是在页面中request.setCharacterEncoding(“GB18030”)来设置请求的编码为中文编码,然后就在每个需要接收参数的页面噼里啪啦的写上这么一句,对当时的水平来说虽然累却也开心。

二:后来开始做分页程序了,突然发现必须得用URL传参了,也就是要用GET方法传参了,那就用吧,程序里照样来request.setCharacterEncoding(“GB18030”)这么一句,然后大失所望,传进来的中文参数(搜索的关键字)出现乱码了,开始郁闷了,查资料才发现原来在 tomcat4 tomcat5 中队参数的处理是不一样的,在 tomcat4 get post的编码是一样的,然而,在 tomcat5 中,get post 的处理是分开进行的。那咋办呢?继续Google,得到的答案是字符转码,来这么一句new String( request.getParameter("para").getBytes("iso8859-1"), "gb18030");发现OK了,高兴!然后遇到类似的中文参数就转啊转的,感觉世界不那么美好了!这还不累死人啊,一会GET转,一会POST转,痛苦!

三:后来,大家开始写一个过滤器,在取得客户端传过来的参数之前,通过过滤器首先将取得的参数编码设定为 gb18030,然后就可以直接使用 getParameter 取得正确的参数了。这个过滤器在 tomcat有示例代码,方便了,感觉世界美好了些,不用每个页面写request.setCharacterEncoding了,但是上面遇到的第二个问题任然没有解决啊,只要是GET方式的中文传参,还得转码。。。先用着吧。。。谁叫Tomcat美国人写的呢。。。

四:不过还好,在 tomcat 5 中,为了解决编码问题,tomcat的作者作了很多努力,具体表现为在 tomcat 的配置文件 server.xml中对 Connector 元素增加了如下的配置参数,专门用来对编码进行直接的配置 URIEncoding=”GB18030”用来设定通过 URI 传递的内容使用的编码,tomcat将使用这里指定的编码对客户端传送的内容进行编码。当然对POST方法还得在过滤器中设置request编码。另一个参数是useBodyEncodingForURI = true设置为true后,只要在页面中request.setCharacterEncoding(“GB18030”)那么无论GET还是POST方式的中文传参都不会出现问题,所以useBodyEncodingForURI = true加过滤器算是一个很好的中文传参解决方法了。

五:上面的方法虽然好,但是对项目的可移植性来说太差了,毕竟咱现在做的些个小项目还没有用专门的web服务器,你让人家服务器就为你一个小屁项目修改自己配置文件,那别人的项目咋办?如果是远程的服务器就更麻烦了。所以还是把这些个处理放到自己的程序中吧。

六:接下来的就是本人现在采用的解决方案,仍然是过滤器,也是本人写这篇文章的中心所在,常规的过滤器只对POST方式有效,所以当请求方式为GET时考虑重写request.getParameter()request.getParameterValues()方法,在自己的方法中将中文转码,既然要重写,而ServletRequest为接口,那么自然得有实现了,我们就继承实现它的类HttpServletRequestWrapper吧,这样外部的JSP/SERVLET就不需要做任何处理了,可谓一劳永逸,呵呵!

好了,还是来看一下具体的代码吧:

过滤器CharsetAllEncodingFilter.java

package filters;

import java.io.IOException;

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.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;

public class CharsetAllEncodingFilter implements Filter {

private String encoding = null;

/**
* Request.java 对 HttpServletRequestWrapper 进行扩充, 不影响原来的功能并能提供所有的
* HttpServletRequest 接口中的功能. 它可以统一的对 Tomcat 默认设置下的中文问题进行解决而只需要用新的 Request
* 对象替换页面中的 request 对象即可.
*/

class Request extends HttpServletRequestWrapper {
  
   private String encoding = null;

   public Request(HttpServletRequest request) {
    super(request);
    encoding = CharsetAllEncodingFilter.this.encoding;
   }

   /**
   * 转换由表单读取的数据的内码. 从 ISO 字符转到 在web.xml中设置的encoding.
   */
   public String toChi(String input) {
    if(input == null) {
     return null;
    } else {
     try {
      byte[] bytes = input.getBytes("ISO8859-1");
      return new String(bytes, encoding);
     } catch (Exception ex) {

     }
     return null;
    }      
   }

   /**
   * Return the HttpServletRequest holded by this object.
   */
   private HttpServletRequest getHttpServletRequest() {
    return (HttpServletRequest) super.getRequest();
   }

   /**
   * 读取参数 -- 修正了中文问题.
   */
   public String getParameter(String name) {
    return toChi(getHttpServletRequest().getParameter(name));
   }

   /**
   * 读取参数列表 - 修正了中文问题.
   */
   public String[] getParameterValues(String name) {
    String values[] = getHttpServletRequest().getParameterValues(name);
    if (values != null) {
     for (int i = 0; i < values.length; i++) {
      values[i] = toChi(values[i]);
     }
    }
    return values;
   }
}

public void destroy() {

}

public void doFilter(ServletRequest request, ServletResponse response,
    FilterChain chain) throws IOException, ServletException {
   HttpServletRequest httpreq = (HttpServletRequest) request;
   if (httpreq.getMethod().equals("POST")) {
    request.setCharacterEncoding(encoding);
   } else {
    request = new Request(httpreq);
   }

   chain.doFilter(request, response);
}

public void init(FilterConfig filterConfig) throws ServletException {
   encoding = filterConfig.getInitParameter("encoding");
}

}

接下来配置web.xml

<filter>

<filter-name>CharsetEncodingFilter</filter-name>

<filter-class>filters.CharsetAllEncodingFilter</filter-class>

<init-param>

      <param-name>encoding</param-name>

      <param-value>GB18030</param-value>

</init-param>

</filter>      

<filter-mapping>

<filter-name>CharsetEncodingFilter</filter-name>

<url-pattern>/*</url-pattern>

</filter-mapping>