前述:
在寫這篇筆記之前,對筆記中的設計模式進行介紹:
本篇筆記中將要使用到的設計模式是:裝飾(包裝)設計模式
(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 }
