應(yīng)用一:解決tomcat下中文亂碼問題(先來個(gè)簡(jiǎn)單的)
在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 {
- //設(shè)置請(qǐng)求響應(yīng)字符編碼
- request.setCharacterEncoding(charset);
- response.setCharacterEncoding(charset);
- HttpServletRequest req = (HttpServletRequest)request;
- System.out.println("----請(qǐng)求被"+config.getFilterName()+"過濾");
- //執(zhí)行下一個(gè)過濾器(如果有的話,否則執(zhí)行目標(biāo)servlet)
- chain.doFilter(req, response);
- System.out.println("----響應(yīng)被"+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中過濾器配置:
- <!--將采用的字符編碼配置成應(yīng)用初始化參數(shù)而不是過濾器私有的初始化參數(shù)是因?yàn)樵贘SP和其他地方也可能需要使用-->
- <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()之前,這樣才能保證參數(shù)是按照已經(jīng)設(shè)置的字符編碼來獲取。
response.setCharacterEncoding(charset);必須寫在PrintWriter out = request.getWriter()之前,這樣才能保證out按照已經(jīng)設(shè)置的字符編碼來進(jìn)行字符輸出。
通過過濾器,我們可以保證在Servlet或JSP執(zhí)行之前就設(shè)置好了請(qǐng)求和響應(yīng)的字符編碼。
但是這樣并不能完全解決中文亂碼問題:
對(duì)于post請(qǐng)求,無論是“獲取參數(shù)環(huán)節(jié)”還是“輸出環(huán)節(jié)"都是沒問題的;
對(duì)于get請(qǐng)求,"輸出環(huán)節(jié)"沒有問題,但是"獲取參數(shù)環(huán)節(jié)"依然出現(xiàn)中文亂碼,所以在輸出時(shí)直接將亂碼輸出了。
原因是post請(qǐng)求和get請(qǐng)求存放參數(shù)位置是不同的:
post方式參數(shù)存放在請(qǐng)求數(shù)據(jù)包的消息體中。get方式參數(shù)存放在請(qǐng)求數(shù)據(jù)包的請(qǐng)求行的URI字段中,以?開始以param=value¶me2=value2的形式附加在URI字段之后。而request.setCharacterEncoding(charset); 只對(duì)消息體中的數(shù)據(jù)起作用,對(duì)于URI字段中的參數(shù)不起作用,我們通常通過下面的代碼來完成編碼轉(zhuǎn)換:
- String paramValue = request.getParameter("paramName");
- paramValue = new String(paramValue.trim().getBytes("ISO-8859-1"), charset);
但是每次進(jìn)行這樣的轉(zhuǎn)換實(shí)在是很麻煩,有沒有統(tǒng)一的解決方案呢?
解決方案1: 在tomcat_home\conf\server.xml 中的Connector元素中設(shè)置URIEncoding屬性為合適的字符編碼
- <Connector port="8080" protocol="HTTP/1.1"
- connectionTimeout="20000"
- redirectPort="8443"
- URIEncoding="UTF-8"
- />
這樣做的缺點(diǎn)是,同一個(gè)tomcat下的其他應(yīng)用也將受到影響。而其每次部署時(shí)都需要類修改配置也很麻煩。
解決方案2:自定義請(qǐng)求包裝器包裝請(qǐng)求,將字符編碼轉(zhuǎn)換的工作添加到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);
- }
- /**
- * 獲得被裝飾對(duì)象的引用和采用的字符編碼
- * @param request
- * @param charset
- */
- public GetHttpServletRequestWrapper(HttpServletRequest request,
- String charset) {
- super(request);
- this.charset = charset;
- }
- /**
- * 實(shí)際上就是調(diào)用被包裝的請(qǐng)求對(duì)象的getParameter方法獲得參數(shù),然后再進(jìn)行編碼轉(zhuǎn)換
- */
- 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("編碼轉(zhuǎn)換之前:" + 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 {
- //設(shè)置請(qǐng)求響應(yīng)字符編碼
- request.setCharacterEncoding(charset);
- response.setCharacterEncoding(charset);
- //新增加的代碼
- HttpServletRequest req = (HttpServletRequest)request;
- if(req.getMethod().equalsIgnoreCase("get"))
- {
- req = new GetHttpServletRequestWrapper(req,charset);
- }
- System.out.println("----請(qǐng)求被"+config.getFilterName()+"過濾");
- //傳遞給目標(biāo)servlet或jsp的實(shí)際上時(shí)包裝器對(duì)象的引用,而不是原始的HttpServletRequest對(duì)象
- chain.doFilter(req, response);
- System.out.println("----響應(yīng)被"+config.getFilterName()+"過濾");
- }
這樣一來,在servlet中調(diào)用包裝器的getParameters方法來獲取參數(shù),就已經(jīng)完成了字符編碼的轉(zhuǎn)換過程,我們就不需要在每次獲取參數(shù)時(shí)來進(jìn)行字符編碼轉(zhuǎn)換了。
其實(shí)用動(dòng)態(tài)代理來實(shí)現(xiàn)可能更好,這里只是我講課時(shí)的一個(gè)例子,用來介紹HttpServletRequestWrapper,不對(duì)之處,敬請(qǐng)指教,以免誤人子弟啊。
源代碼示例工程:http://www.aygfsteel.com/Files/whistler/demo.rar
學(xué)軟件開發(fā),到蜂鳥科技!
超強(qiáng)的師資力量 、完善的課程體系 、超低的培訓(xùn)價(jià)格 、真實(shí)的企業(yè)項(xiàng)目。
網(wǎng)址:
電話:
鄭州軟件開發(fā)興趣小組群: