使用Decorate模式實現留言版詞匯處理
裝飾者模式以對客戶端透明的方式動態的為對象增加責任。此模式提供了一個比繼承更為靈活的替代方案來擴展對象的功能,避免了繼承方法產生的類激增問題,而且更方便更改對象的責任。
我們經常要為某一些個別的對象增加一些新的職責,并不是全部的類。例如我們系統留言反饋板塊中可能需要過濾用戶輸入留言中的一些詞匯(例如政治敏感詞匯、色情詞匯等)、還可能對用戶輸入留言進行一些修飾(例如對用戶輸入的URL自動加上超鏈接、對用戶輸入的UBB代碼進行轉換的)、還可能將用戶輸入的內容定時發送的網管的郵箱中等等。如果使用類繼承的方式進行設計,我們可能要設計一個接口
BodyContentFilterIntf,然后在由BodyContentFilterIntf派生出SensitiveWordContentFilter、HtmlContentFilter、SendEmailContentFilter等類。但是如果還要要求同時能過濾敏感詞匯并能進行修飾、或者過濾敏感詞匯之后把用戶輸入的留言發送到網管郵箱等等,這樣就要增加SensitiveWordHtmlContentFilter、SensitiveWordSendEmaillContentFilter等類,這種方式導致了子類瀑發式的產生。
一個靈活的方法是將過濾器嵌入另一個過濾器中,由這個過濾器來負責調用被嵌入過濾器的方法并執行自己的過濾器方法。我們稱這個嵌入的過濾器為裝飾(Decorator)。這個裝飾與過濾器接口一致。裝飾將請求向前轉到到另一個過濾器,并且可能能轉發前后執行一些額外的動作(如修飾、發送郵件),透明性使你可以遞歸的嵌套多個裝飾,從面可以添加任意多的功能。
其實java中的過濾器模式應用非常多,典型的就是IO的Stream操作。在IO處理中,Java將數據抽象為流(Stream)。在IO庫中,最基本的是InputStream和OutputStream兩個分別處理輸出和輸入的對象,但是在InputStream和OutputStream中之提供了最簡單的流處理方法,只能讀入/寫出字符,沒有緩沖處理,無法處理文件,等等。
LineNumberInputStream、BufferInputStream、StringBufferInputStream等提供各種不同服務的類只要組合起來就可以實現很多功能,如下:
多個的Decorator被層疊在一起,最后得到一個功能強大的流。既能夠被緩沖,又能夠得到行數,這就是Decorator的威力!
下面是我們的類靜態圖
我們定義一個接口BodyContentFilterIntf 來定義所有過濾器要實現的方法:
這個接口中只有一個方法filtContent,將要過濾的留言傳給aContent參數,filtContent對aContent進行一些處理(如裝飾URL、UBB等),然后將處理后的字符串做為返回值返回;如果留言沒有通過過濾(如含有敏感詞匯等),只要拋出自定義ContentFilterException異常即可。
下面是一個可能的一個過濾器(保證輸入的字數多于50):
這是另一個過濾器(偽碼,用來實現向網管郵箱發送郵件) public class SendEmailContentFilter
當然還有SensitiveWordContextFilter(過濾敏感詞匯),HtmlContentFilter(修飾用戶輸入留言中的超級鏈接)等。
有了這些過濾器,我們就可以很方便的為留言版添加各種復合的過濾器。例如我們想對輸入的留言進行超鏈接修飾和過濾敏感詞匯,那么我們只要如下調用即可:
我們甚至可以動態的添加不同的過濾器,例如對于會員我們要對輸入的留言進行超鏈接修飾并且將他的留言發送到網管郵箱,而對于非會員我們則要過濾他輸入的敏感詞匯并且保證輸入的字數不少于50,我們只要如下調用即可:
我們經常要為某一些個別的對象增加一些新的職責,并不是全部的類。例如我們系統留言反饋板塊中可能需要過濾用戶輸入留言中的一些詞匯(例如政治敏感詞匯、色情詞匯等)、還可能對用戶輸入留言進行一些修飾(例如對用戶輸入的URL自動加上超鏈接、對用戶輸入的UBB代碼進行轉換的)、還可能將用戶輸入的內容定時發送的網管的郵箱中等等。如果使用類繼承的方式進行設計,我們可能要設計一個接口
BodyContentFilterIntf,然后在由BodyContentFilterIntf派生出SensitiveWordContentFilter、HtmlContentFilter、SendEmailContentFilter等類。但是如果還要要求同時能過濾敏感詞匯并能進行修飾、或者過濾敏感詞匯之后把用戶輸入的留言發送到網管郵箱等等,這樣就要增加SensitiveWordHtmlContentFilter、SensitiveWordSendEmaillContentFilter等類,這種方式導致了子類瀑發式的產生。
一個靈活的方法是將過濾器嵌入另一個過濾器中,由這個過濾器來負責調用被嵌入過濾器的方法并執行自己的過濾器方法。我們稱這個嵌入的過濾器為裝飾(Decorator)。這個裝飾與過濾器接口一致。裝飾將請求向前轉到到另一個過濾器,并且可能能轉發前后執行一些額外的動作(如修飾、發送郵件),透明性使你可以遞歸的嵌套多個裝飾,從面可以添加任意多的功能。
其實java中的過濾器模式應用非常多,典型的就是IO的Stream操作。在IO處理中,Java將數據抽象為流(Stream)。在IO庫中,最基本的是InputStream和OutputStream兩個分別處理輸出和輸入的對象,但是在InputStream和OutputStream中之提供了最簡單的流處理方法,只能讀入/寫出字符,沒有緩沖處理,無法處理文件,等等。
LineNumberInputStream、BufferInputStream、StringBufferInputStream等提供各種不同服務的類只要組合起來就可以實現很多功能,如下:
FilterInputStream myStream=new LineNumberInputStream ( new BufferInputStream( new StringBufferInputStream( myStringBuffer))); |
多個的Decorator被層疊在一起,最后得到一個功能強大的流。既能夠被緩沖,又能夠得到行數,這就是Decorator的威力!
下面是我們的類靜態圖
我們定義一個接口BodyContentFilterIntf 來定義所有過濾器要實現的方法:
public interface BodyContentFilterIntf { public String filtContent(String aContent) throws ContentFilterException; } |
這個接口中只有一個方法filtContent,將要過濾的留言傳給aContent參數,filtContent對aContent進行一些處理(如裝飾URL、UBB等),然后將處理后的字符串做為返回值返回;如果留言沒有通過過濾(如含有敏感詞匯等),只要拋出自定義ContentFilterException異常即可。
下面是一個可能的一個過濾器(保證輸入的字數多于50):
public class LengthContentFilter implements BodyContentFilterIntf { private BodyContentFilterIntf bodyContentFilterIntf = null; public HtmlContentFilter(BodyContentFilterIntf aFilter) { bodyContentFilterIntf = aFilter; } public String filtContent(String aContent) throws ContentFilterException { String l_Content = aContent; If (bodyContentFilterIntf!=null) _Content = bodyContentFilterIntf .filtContent(l_Content); if (aContent.length()<=50) throw new ContentFilterException (“輸入的字數不能少于50!”); return aContext; } } |
這是另一個過濾器(偽碼,用來實現向網管郵箱發送郵件) public class SendEmailContentFilter
implements BodyContentFilterIntf { private BodyContentFilterIntf bodyContentFilterIntf = null; public SendEmailContentFilter(BodyContentFilterIntf aFilter) { bodyContentFilterIntf = aFilter; } public String filtContent(String aContent) throws ContentFilterException { String l_Content = aContent; if (bodyContentFilterIntf!=null) l_Content = bodyContentFilterIntf .filtContent(l_Content); SendEmail(“webmaster@SnailWeb.com”,l_Content) return aContext; } } |
當然還有SensitiveWordContextFilter(過濾敏感詞匯),HtmlContentFilter(修飾用戶輸入留言中的超級鏈接)等。
有了這些過濾器,我們就可以很方便的為留言版添加各種復合的過濾器。例如我們想對輸入的留言進行超鏈接修飾和過濾敏感詞匯,那么我們只要如下調用即可:
try { l_Content = new HtmlContentFilter(new SensitiveWordContextFilter(null)). filtContent(bodyContext); } catch (ContentFilterException ex) { BBSCommon.showMsgInResponse(response, ex.getMessage()); return; } |
我們甚至可以動態的添加不同的過濾器,例如對于會員我們要對輸入的留言進行超鏈接修飾并且將他的留言發送到網管郵箱,而對于非會員我們則要過濾他輸入的敏感詞匯并且保證輸入的字數不少于50,我們只要如下調用即可:
try { BodyContentFilterIntf bodyContentFilterIntf = null; bodyContentFilterIntf = new HtmlContentFilter(null); if(IsMember==true) bodyContentFilterIntf = new sendEmailContentFilter(bodyContentFilterIntf); else bodyContentFilterIntf = new SensitiveWordContextFilter(bodyContentFilterIntf); l_Content = bodyContentFilterIntf.filtContent(bodyContext); } catch (ContentFilterException ex) { BBSCommon.showMsgInResponse(response, ex.getMessage()); return; } |