Spring-涉及到的設(shè)計(jì)模式匯總
1. 簡(jiǎn)單工廠 又叫做靜態(tài)工廠方法(StaticFactory Method)模式,但不屬于23種GOF設(shè)計(jì)模式之一。 簡(jiǎn)單工廠模式的實(shí)質(zhì)是由一個(gè)工廠類根據(jù)傳入的參數(shù)卖陵,動(dòng)態(tài)決定應(yīng)該創(chuàng)建哪一個(gè)產(chǎn)品類虚吟。 Spring中的BeanFactory就是簡(jiǎn)單工廠模式的體現(xiàn)画髓,根據(jù)傳入一個(gè)唯一的標(biāo)識(shí)來(lái)獲得Bean對(duì)象,但是否是在傳入?yún)?shù)后創(chuàng)建還是傳入?yún)?shù)前創(chuàng)建這個(gè)要根據(jù)具體情況來(lái)定垦梆。
2. 工廠方法(Factory Method) 定義一個(gè)用于創(chuàng)建對(duì)象的接口,讓子類決定實(shí)例化哪一個(gè)類仅孩。Factory Method使一個(gè)類的實(shí)例化延遲到其子類托猩。 Spring中的FactoryBean就是典型的工廠方法模式。如下圖:
?3. 單例(Singleton) 保證一個(gè)類僅有一個(gè)實(shí)例辽慕,并提供一個(gè)訪問(wèn)它的全局訪問(wèn)點(diǎn)京腥。 Spring中的單例模式完成了后半句話,即提供了全局的訪問(wèn)點(diǎn)BeanFactory溅蛉。但沒有從構(gòu)造器級(jí)別去控制單例公浪,這是因?yàn)镾pring管理的是是任意的Java對(duì)象。
?4. 適配器(Adapter) 將一個(gè)類的接口轉(zhuǎn)換成客戶希望的另外一個(gè)接口船侧。Adapter模式使得原本由于接口不兼容而不能一起工作的那些類可以一起工作欠气。 Spring中在對(duì)于AOP的處理中有Adapter模式的例子,見如下圖: 由于Advisor鏈需要的是MethodInterceptor(攔截器)對(duì)象镜撩,所以每一個(gè)Advisor中的Advice都要適配成對(duì)應(yīng)的MethodInterceptor對(duì)象预柒。
?5.包裝器(Decorator) 動(dòng)態(tài)地給一個(gè)對(duì)象添加一些額外的職責(zé)。就增加功能來(lái)說(shuō)琐鲁,Decorator模式相比生成子類更為靈活卫旱。 Spring中用到的包裝器模式在類名上有兩種表現(xiàn):一種是類名中含有Wrapper,另一種是類名中含有Decorator围段」艘恚基本上都是動(dòng)態(tài)地給一個(gè)對(duì)象添加一些額外的職責(zé)。 6. 代理(Proxy) 為其他對(duì)象提供一種代理以控制對(duì)這個(gè)對(duì)象的訪問(wèn)奈泪。 從結(jié)構(gòu)上來(lái)看和Decorator模式類似适贸,但Proxy是控制,更像是一種對(duì)功能的限制涝桅,而Decorator是增加職責(zé)拜姿。 Spring的Proxy模式在aop中有體現(xiàn),比如JdkDynamicAopProxy和Cglib2AopProxy冯遂。 7.觀察者(Observer) 定義對(duì)象間的一種一對(duì)多的依賴關(guān)系蕊肥,當(dāng)一個(gè)對(duì)象的狀態(tài)發(fā)生改變時(shí),所有依賴于它的對(duì)象都得到通知并被自動(dòng)更新。 Spring中Observer模式常用的地方是listener的實(shí)現(xiàn)壁却。如ApplicationListener批狱。
?8. 策略(Strategy) 定義一系列的算法,把它們一個(gè)個(gè)封裝起來(lái)展东,并且使它們可相互替換赔硫。本模式使得算法可獨(dú)立于使用它的客戶而變化。 Spring中在實(shí)例化對(duì)象的時(shí)候用到Strategy模式盐肃,見如下圖: 在SimpleInstantiationStrategy中有如下代碼說(shuō)明了策略模式的使用情況:
?9.模板方法(Template Method) 定義一個(gè)操作中的算法的骨架爪膊,而將一些步驟延遲到子類中。Template Method使得子類可以不改變一個(gè)算法的結(jié)構(gòu)即可重定義該算法的某些特定步驟砸王。 Template Method模式一般是需要繼承的推盛。這里想要探討另一種對(duì)Template Method的理解。Spring中的JdbcTemplate谦铃,在用這個(gè)類時(shí)并不想去繼承這個(gè)類小槐,因?yàn)檫@個(gè)類的方法太多,但是我們還是想用到JdbcTemplate已有的穩(wěn)定的荷辕、公用的數(shù)據(jù)庫(kù)連接凿跳,那么我們?cè)趺崔k呢?我們可以把變化的東西抽出來(lái)作為一個(gè)參數(shù)傳入JdbcTemplate的方法中疮方。但是變化的東西是一段代碼控嗜,而且這段代碼會(huì)用到JdbcTemplate中的變量。怎么辦骡显?那我們就用回調(diào)對(duì)象吧疆栏。在這個(gè)回調(diào)對(duì)象中定義一個(gè)操縱JdbcTemplate中變量的方法,我們?nèi)?shí)現(xiàn)這個(gè)方法惫谤,就把變化的東西集中到這里了壁顶。然后我們?cè)賯魅脒@個(gè)回調(diào)對(duì)象到JdbcTemplate,從而完成了調(diào)用溜歪。這可能是Template Method不需要繼承的另一種實(shí)現(xiàn)方式吧若专。 以下是一個(gè)具體的例子: JdbcTemplate中的execute方法: JdbcTemplate執(zhí)行execute方法: