筆記

          way

          HttpServletRequestWrapper相關(guān)

          應(yīng)用一:解決tomcat下中文亂碼問(wèn)題(先來(lái)個(gè)簡(jiǎn)單的) 

          tomcat下,我們通常這樣來(lái)解決中文亂碼問(wèn)題:

          過(guò)濾器代碼:

          1. package filter;  
          2.   
          3. import java.io.*;  
          4. import javax.servlet.*;  
          5. import javax.servlet.http.*;  
          6. import wrapper.GetHttpServletRequestWrapper;  
          7.   
          8. public class ContentTypeFilter implements Filter {  
          9.   
          10.     private String charset = "UTF-8";  
          11.     private FilterConfig config;  
          12.       
          13.     public void destroy() {  
          14.         System.out.println(config.getFilterName()+"被銷(xiāo)毀");  
          15.         charset = null;  
          16.         config = null;  
          17.     }  
          18.   
          19.     public void doFilter(ServletRequest request, ServletResponse response,  
          20.             FilterChain chain) throws IOException, ServletException {  
          21.         //設(shè)置請(qǐng)求響應(yīng)字符編碼  
          22.         request.setCharacterEncoding(charset);  
          23.         response.setCharacterEncoding(charset);  
          24.           
          25.         HttpServletRequest req = (HttpServletRequest)request;  
          26.           
          27.           
          28.         System.out.println("----請(qǐng)求被"+config.getFilterName()+"過(guò)濾");  
          29.         //執(zhí)行下一個(gè)過(guò)濾器(如果有的話,否則執(zhí)行目標(biāo)servlet)  
          30.         chain.doFilter(req, response);  
          31.           
          32.         System.out.println("----響應(yīng)被"+config.getFilterName()+"過(guò)濾");  
          33.   
          34.     }  
          35.   
          36.     public void init(FilterConfig config) throws ServletException {  
          37.             this.config = config;  
          38.             String charset = config.getServletContext().getInitParameter("charset");    
          39.             if( charset != null && charset.trim().length() != 0)  
          40.             {  
          41.                 this.charset = charset;  
          42.             }  
          43.     }  
          44.   
          45. }  

          web.xml中過(guò)濾器配置
          <!--將采用的字符編碼配置成應(yīng)用初始化參數(shù)而不是過(guò)濾器私有的初始化參數(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>

          equest.setCharacterEncoding(charset); 必須寫(xiě)在第一次使用request.getParameter()之前,這樣才能保證參數(shù)是按照已經(jīng)設(shè)置的字符編碼來(lái)獲取。
          response.setCharacterEncoding(charset);必須寫(xiě)在PrintWriter out = request.getWriter()之前,這樣才能保證out按照已經(jīng)設(shè)置的字符編碼來(lái)進(jìn)行字符輸出。

          通過(guò)過(guò)濾器,我們可以保證在Servlet或JSP執(zhí)行之前就設(shè)置好了請(qǐng)求和響應(yīng)的字符編碼。

          但是這樣并不能完全解決中文亂碼問(wèn)題:

          對(duì)于post請(qǐng)求,無(wú)論是“獲取參數(shù)環(huán)節(jié)”還是“輸出環(huán)節(jié)"都是沒(méi)問(wèn)題的;

          對(duì)于get請(qǐng)求,"輸出環(huán)節(jié)"沒(méi)有問(wèn)題,但是"獲取參數(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字段中,以?開(kāi)始以param=value&parame2=value2的形式附加在URI字段之后。而request.setCharacterEncoding(charset); 只對(duì)消息體中的數(shù)據(jù)起作用,對(duì)于URI字段中的參數(shù)不起作用,我們通常通過(guò)下面的代碼來(lái)完成編碼轉(zhuǎn)換:

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

          但是每次進(jìn)行這樣的轉(zhuǎn)換實(shí)在是很麻煩,有沒(méi)有統(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í)都需要類(lèi)修改配置也很麻煩。

          解決方案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;
                  }

              }

          }


          修改過(guò)濾器的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() + "過(guò)濾");
                  
          // 傳遞給目標(biāo)servlet或jsp的實(shí)際上時(shí)包裝器對(duì)象的引用,而不是原始的HttpServletRequest對(duì)象
                  chain.doFilter(req, response);

                  System.out.println(
          "----響應(yīng)被" + config.getFilterName() + "過(guò)濾");
              }

           

           

          這樣一來(lái),在servlet中調(diào)用包裝器的getParameters方法來(lái)獲取參數(shù),就已經(jīng)完成了字符編碼的轉(zhuǎn)換過(guò)程,我們就不需要在每次獲取參數(shù)時(shí)來(lái)進(jìn)行字符編碼轉(zhuǎn)換了。
          總結(jié):自己寫(xiě)類(lèi)繼承HttpServletRequestWrapper,HttpServletRequestWrapper實(shí)現(xiàn)了HttpServletRequest接口。看tomcat的源代碼可以發(fā)現(xiàn),ServletRequest作為一個(gè)Component ,ServletRequestWrapper作為一個(gè)比較標(biāo)準(zhǔn)的Decorator ,實(shí)現(xiàn)ServletRequest接口并把ServletRequest當(dāng)作成員變量,其他繼承Decorator 的類(lèi)(比如本例中的GetHttpServletRequestWrapper )就可以很好的操控ServletRequest及其子類(lèi)(比如本例中的HttpServletRequest),HttpServletRequest的很多方法就可以根據(jù)我們的需求做改變,比如設(shè)置字符,去掉空格。
          參考:
          http://www.javaeye.com/topic/483158
          http://fishhappy365.javaeye.com/blog/484185
          http://www.javaeye.com/topic/220230

          posted on 2009-11-22 22:30 yuxh 閱讀(7205) 評(píng)論(0)  編輯  收藏 所屬分類(lèi): j2ee設(shè)計(jì)模式

          導(dǎo)航

          <2009年11月>
          25262728293031
          1234567
          891011121314
          15161718192021
          22232425262728
          293012345

          統(tǒng)計(jì)

          常用鏈接

          留言簿

          隨筆分類(lèi)

          隨筆檔案

          收藏夾

          博客

          搜索

          最新評(píng)論

          閱讀排行榜

          評(píng)論排行榜

          主站蜘蛛池模板: 鹤峰县| 三河市| 丰城市| 玉树县| 苍南县| 名山县| 宁安市| 龙胜| 都昌县| 乌恰县| 东明县| 益阳市| 马尔康县| 鄂托克前旗| 恩平市| 星座| 溧阳市| 东光县| 富蕴县| 静宁县| 琼海市| 蕲春县| 永康市| 霍城县| 简阳市| 安新县| 横山县| 旌德县| 广河县| 大安市| 湄潭县| 藁城市| 年辖:市辖区| 黄梅县| 屏东市| 大安市| 宁津县| 天峨县| 巴青县| 沙湾县| 鄂温|