设为首页 收藏本站
查看: 496|回复: 0

[经验分享] 通过HttpServletRequestWrapper解决Tomcat请求乱码问题

[复制链接]

尚未签到

发表于 2017-1-31 12:39:59 | 显示全部楼层 |阅读模式
应用一:解决tomcat下中文乱码问题(先来个简单的)
在tomcat下,我们通常这样来解决中文乱码问题:
过滤器代码:

package filter;  
import java.io.*;  
import javax.servlet.*;  
import javax.servlet.http.*;  
import wrapper.GetHttpServletRequestWrapper;  
public class ContentTypeFilter implements Filter {  
private String charset = "UTF-8";  
private FilterConfig config;  
public void destroy() {  
System.out.println(config.getFilterName()+"被销毁");  
charset = null;  
config = null;  
}  
public void doFilter(ServletRequest request, ServletResponse response,  
FilterChain chain) throws IOException, ServletException {  
//设置请求响应字符编码  
request.setCharacterEncoding(charset);  
response.setCharacterEncoding(charset);  
HttpServletRequest req = (HttpServletRequest)request;  

System.out.println("----请求被"+config.getFilterName()+"过滤");  
//执行下一个过滤器(如果有的话,否则执行目标servlet)  
chain.doFilter(req, response);  
System.out.println("----响应被"+config.getFilterName()+"过滤");  
}  
public void init(FilterConfig config) throws ServletException {  
this.config = config;  
String charset = config.getServletContext().getInitParameter("charset");   
if( charset != null && charset.trim().length() != 0)  
{  
this.charset = charset;  
}  
}  
}  

web.xml中过滤器配置:

<!--将采用的字符编码配置成应用初始化参数而不是过滤器私有的初始化参数是因为在JSP和其他地方也可能需要使用-->  
<context-param>  
<param-name>charset</param-name>  
<param-value>UTF-8</param-value>  
</context-param>  
<filter>  
<filter-name>ContentTypeFilter</filter-name>  
<filter-class>filter.ContentTypeFilter</filter-class>  
</filter>  
<filter-mapping>  
<filter-name>ContentTypeFilter</filter-name>  
<url-pattern>/*</url-pattern>  
</filter-mapping>  

request.setCharacterEncoding(charset); 必须写在第一次使用request.getParameter()之前,这样才能保证参数是按照已经设置的字符编码来获取。
response.setCharacterEncoding(charset);必须写在PrintWriter out = request.getWriter()之前,这样才能保证out按照已经设置的字符编码来进行字符输出。
通过过滤器,我们可以保证在Servlet或JSP执行之前就设置好了请求和响应的字符编码。
但是这样并不能完全解决中文乱码问题:
对于post请求,无论是“获取参数环节”还是“输出环节"都是没问题的;
对于get请求,"输出环节"没有问题,但是"获取参数环节"依然出现中文乱码,所以在输出时直接将乱码输出了。
原因是post请求和get请求存放参数位置是不同的:
post方式参数存放在请求数据包的消息体中。get方式参数存放在请求数据包的请求行的URI字段中,以?开始以param=value&parame2=value2的形式附加在URI字段之后。而request.setCharacterEncoding(charset); 只对消息体中的数据起作用,对于URI字段中的参数不起作用,我们通常通过下面的代码来完成编码转换:

String paramValue = request.getParameter("paramName");  
paramValue = new String(paramValue.trim().getBytes("ISO-8859-1"), charset);

但是每次进行这样的转换实在是很麻烦,有没有统一的解决方案呢?
解决方案1: 在tomcat_home\conf\server.xml 中的Connector元素中设置URIEncoding属性为合适的字符编码

<Connector port="8080" protocol="HTTP/1.1"   
connectionTimeout="20000"   
redirectPort="8443"   
URIEncoding="UTF-8"  
/>  

解决方案2:自定义请求包装器包装请求,将字符编码转换的工作添加到getParameter()方法中

package wrapper;  
import java.io.UnsupportedEncodingException;  
import java.net.URLDecoder;  
import javax.servlet.http.HttpServletRequest;  
import javax.servlet.http.HttpServletRequestWrapper;  
public class GetHttpServletRequestWrapper extends HttpServletRequestWrapper {  
private String charset = "UTF-8";  
public GetHttpServletRequestWrapper(HttpServletRequest request) {  
super(request);  
}  
/**
* 获得被装饰对象的引用和采用的字符编码
* @param request
* @param charset
*/  
public GetHttpServletRequestWrapper(HttpServletRequest request,  
String charset) {  
super(request);  
this.charset = charset;  
}  
/**
* 实际上就是调用被包装的请求对象的getParameter方法获得参数,然后再进行编码转换
*/  
public String getParameter(String name) {  
String value = super.getParameter(name);  
value = value == null ? null : convert(value);  
return value;  
}  
public String convert(String target) {  
System.out.println("编码转换之前:" + target);  
try {  
return new String(target.trim().getBytes("ISO-8859-1"), charset);  
} catch (UnsupportedEncodingException e) {  
return target;  
}  
}  
}  

修改过滤器的doFilter方法 代码如下:

public void doFilter(ServletRequest request, ServletResponse response,  
FilterChain chain) throws IOException, ServletException {  
//设置请求响应字符编码  
request.setCharacterEncoding(charset);  
response.setCharacterEncoding(charset);  
//新增加的代码         
HttpServletRequest req = (HttpServletRequest)request;  
if(req.getMethod().equalsIgnoreCase("get"))  
{  
req = new GetHttpServletRequestWrapper(req,charset);  
}  
System.out.println("----请求被"+config.getFilterName()+"过滤");  
//传递给目标servlet或jsp的实际上时包装器对象的引用,而不是原始的HttpServletRequest对象  
chain.doFilter(req, response);  
System.out.println("----响应被"+config.getFilterName()+"过滤");  
}

这样一来,在servlet中调用包装器的getParameters方法来获取参数,就已经完成了字符编码的转换过程,我们就不需要在每次获取参数时来进行字符编码转换了。
原文出处:http://www.iteye.com/topic/483158

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-335739-1-1.html 上篇帖子: Tomcat 6下创建虚拟目录 下篇帖子: tomcat服务器开启gzip压缩功能
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表