使用過濾器(Filter)解決請求參數中文亂碼問題(復雜方式)


前述:
     在寫這篇筆記之前,對筆記中的設計模式進行介紹:
     本篇筆記中將要使用到的設計模式是:裝飾(包裝)設計模式
          (1)裝飾(包裝)設計模式口訣:
               ①定義一個類,實現被裝飾對象的接口
               ②定義一個成員變量,記住被裝飾對象的引用
               ③定義構造方法,傳入被裝飾對象的實例
               ④改寫要修改的方法
               ⑤不需要改寫的方法,調用被裝飾對象的原來的方法
 
          (2)什么時候使用裝飾設計模式
               當我們需要對一個類進行增強的時候,增強后的類不再當前類的范疇
                    例如:現在有一個     Animal類     Cat和Dog都屬於動物類型,因此可以直接繼承
                                                                      現在新來一個“電子狗 ”,不屬於動物的范圍,但是有需要用其中的方法,這時候我們選擇使用裝飾(包裝)設計模式
 
一:需求:統一解決請求參數中文亂碼
  
二:需求原因:
     在沒有該需求之前,解決請求亂碼問題:
          (1)POST:
 
 1 //第一種
 2 //request.setCharacterEncoding(this.getServletContext().getInitParameter("charset"));
 3 //備注:這種獲取方式是因為在web.xml中進行了如下配置
 4      <!-- 設置編碼 -->
 5       <context-param>
 6             <param-name>charset</param-name>
 7             <param-value>UTF-8</param-value>
 8       </context-param>
 9  
10 //第二種
11 request.setCharacterEncoding("utf-8");

 

     分析:第一種方式的好處是,在context中配置后,項目后期如果需要修改獲取方式,直接修改配置文件即可,不要對.java文件進行修改
 
         (2)GET:
 
1      String value = request.getParameter("value");
2      if(value == null || value.trim().equals("")){
3          value="";
4      }
5      value = new String(value.getBytes("ISO-8859-1"),"utf-8");

 

     
三、優化思路:
     使用一個過濾器,在請求到達servlet之前,先對request對象設置編碼
     要求所有的請求都要進行設置編碼,因此所有的request都要攔截,進行增強,那么:
 
1      <filter>
2             <filter-name>EncodingFilter</filter-name>
3             <filter-class>com.cqy.filter.EncodingFilter</filter-class>
4       </filter>
5  
6       <filter-mapping>
7             <filter-name>EncodingFilter</filter-name>
8             <url-pattern>/*</url-pattern>
9       </filter-mapping>

 

 
四:代碼實現:
     1、過濾器代碼
 
 1 package com.cqy.filter;
 2  
 3 import java.io.IOException;
 4  
 5 import javax.servlet.Filter;
 6 import javax.servlet.FilterChain;
 7 import javax.servlet.FilterConfig;
 8 import javax.servlet.ServletException;
 9 import javax.servlet.ServletRequest;
10 import javax.servlet.ServletResponse;
11 import javax.servlet.http.HttpServletRequest;
12 import javax.servlet.http.HttpServletResponse;
13  
14 import com.cqy.domain.MyRequest;
15  
16 public class EncodingFilter implements Filter {
17  
18       @Override
19       public void destroy() {
20  
21       }
22  
23       @Override
24       public void doFilter(ServletRequest req, ServletResponse res,
25                   FilterChain chain) throws IOException, ServletException {
26             // 將請求和響應強制轉換成Http形式
27             HttpServletRequest request = (HttpServletRequest) req;
28             HttpServletResponse response = (HttpServletResponse) res;
29  
30             // 處理響應亂碼
31             response.setContentType("text/html;charset=utf-8");
32  
33             // 自定義一個request對象:MyRequest,對服務器原來的requset進行增強,使用裝飾設計模式
34             // 要增強原來的request對象,必須先獲取到原來的request對象
35             MyRequest myRequest = new MyRequest(request);
36  
37             // 注意:放行的時候應該傳入增強后的request對象
38             chain.doFilter(myRequest, response);
39       }
40  
41       @Override
42       public void init(FilterConfig arg0) throws ServletException {
43  
44       }
45  
46 }

 

 
     2、自定義增強類(MyRequest )
 
  1 package com.domain;
  2  
  3 import java.io.UnsupportedEncodingException;
  4 import java.util.Map;
  5 import java.util.Set;
  6  
  7 import javax.servlet.http.HttpServletRequest;
  8 import javax.servlet.http.HttpServletRequestWrapper;
  9  
 10 /**
 11  * @author  繼承HttpServletRequestWrapper相當於實現了HttpServletRequest
 12  *         HttpServletRequestWrapper類,它本身實現了所有HttpServletRequest的方法
 13  *         繼承它之后,需要修改的方法MyRequest可以自己定義,不需要修改的方法,直接使用父類的方法
 14  *
 15  *         第一步:總結:繼承HttpServletRequestWrapper,為了偷懶,
 16  *         不用自己去實現所有HttpServletRequest的方法 第二步:使用構造函數將原來的request對象保存到當前自定義對象中
 17  *         第三步:針對要修改的方法,進行增強 第四步:定義一個flag標記,防止編碼重復執行
 18  */
 19 public class MyRequest extends HttpServletRequestWrapper {
 20  
 21     // 定義了一個成員變量,用來保存構造函數傳入的requset對象
 22     private HttpServletRequest request = null;
 23  
 24     // 定義一個標記,用來標注:當前requset中,請求參數,是否已經編碼過了
 25     private boolean flag = false;
 26  
 27     public MyRequest(HttpServletRequest request) {
 28         super(request);
 29         this.request = request;
 30  
 31     }
 32  
 33     // 總需求:對request對象的獲取數據的方法,進行增強(統一編碼)
 34  
 35     @Override
 36     public Map<String, String[]> getParameterMap() {
 37         // 獲得請求方式request.getMethod()方法
 38         String method = this.request.getMethod();
 39         // post請求
 40         if ("post".equalsIgnoreCase(method)) {
 41             // 設置編碼格式
 42             try {
 43                 request.setCharacterEncoding("utf-8");
 44             } catch (UnsupportedEncodingException e) {
 45                 e.printStackTrace();
 46             }
 47             Map<String, String[]> map = this.request.getParameterMap();
 48             return map;
 49  
 50         } else if ("get".equalsIgnoreCase(method)) {
 51             // get請求
 52             // 分析:get請求需要對每一個參數都進行轉換,因此需要對map中的每個元素進行遍歷
 53             // 首先獲得map集合
 54             Map<String, String[]> map = this.request.getParameterMap();
 55  
 56             //第一次獲取請求參數,flag==false,執行后面的額亂碼處理動作
 57             //第二次獲取請求參數的時候,flag==true,不執行后面的處理,直接返回已經編碼過的map集合
 58             if (flag) {
 59                 return map;
 60             }
 61             if (map == null) {
 62                 return super.getParameterMap();
 63             } else {
 64                 // 然后獲得map集合的key
 65                 Set<String> key = map.keySet();
 66                 // 通過key將map中的元素取出來
 67                 for (String string : key) {
 68                     String[] value = map.get(string);
 69                     // 接下來需要將String中的每一個都進行遍歷,轉換參數
 70                     for (int i = 0; i < value.length; i++) {
 71                         try {
 72                             String string2 = new String(
 73                                     value[i].getBytes("iso-8859-1"), "utf-8");
 74                             value[i] = string2;
 75                         } catch (UnsupportedEncodingException e) {
 76                             e.printStackTrace();
 77                         }
 78                     }
 79                 }
 80                 flag = true;
 81                 return map;
 82             }
 83         } else {
 84             //位置請求方式,自定義對象處理不了,使用父類的方法處理
 85             return super.getParameterMap();
 86         }
 87     }
 88  
 89     @Override
 90     public String[] getParameterValues(String name) {
 91         // 通過map集合獲取參數
 92         Map<String, String[]> map = this.getParameterMap();
 93         if (map == null) {
 94             return super.getParameterValues(name);
 95         } else {
 96             String[] strings = map.get(name);
 97             return strings;
 98         }
 99     }
100  
101     @Override
102     public String getParameter(String name) {
103         // 通過values獲取參數
104         String[] values = this.getParameterValues(name);
105         if (values == null) {
106             return super.getParameter(name);
107         } else {
108             return values[0];
109         }
110     }
111  
112 }

 

 
 
 
   


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM