Java代理深入講解之靜態(tài)代理
什么是代理
代理就是給目標對象一個代理對象,并由代理對象控制目標的引用。
為什么要使用代理模式
1、通過引入代理對象的方式,可以間接的訪問目標對象,避免直接訪問目標對象給系統(tǒng)帶來不必要的復雜性。
2、通過代理對象可以對原有的業(yè)務進行業(yè)務增強處理。
舉例:如果我們需要買國外的某一件商品A,這個時候我們一般有兩個途徑要么直接去國外買,要么可以找一些代購人員幫我們?nèi)ベ徺I。在這種情況下,我們由于直接去國外買,實在是太耗軟妹幣,而且還要花時間等等,這個時候我們最優(yōu)的選擇就是找代購購買,這樣也幫我們省去了很多麻煩的事情。
代理模式類圖
代碼示例
抽象對象:
public interface ITargetFactoryService { void sale(String name); }
目標對象:
@Slf4j public class TargetFactoryServiceImpl implements ITargetFactoryService { @Override public void sale(String name) { log.info(name+"購買了商品A"); } }
代理對象:
@Slf4j public class ProxyImpl implements ITargetFactoryService { public ITargetFactoryService service; public ProxyImpl(ITargetFactoryService service){ super(); this.service = service; } @Override public void sale(String name) { before(); service.sale("代購"); after(); } /** * 后置增強 */ private void after() { log.info("代購在購買后得到了市場調(diào)研結(jié)果"); } /** * 前置增強 */ private void before() { log.info("代購在購買前做了市場調(diào)研"); } }
測試類:
@Slf4j @SpringBootApplication(exclude = {DataSourceAutoConfiguration.class, DataSourceTransactionManagerAutoConfiguration.class, JdbcTemplateAutoConfiguration.class}) public class SpsringJdbcApplication { public static void main(String[] args) { TargetFactoryServiceImpl service = new TargetFactoryServiceImpl(); ProxyImpl proxy = new ProxyImpl(service); proxy.sale("代購"); SpringApplication.run(SpsringJdbcApplication.class, args); } }
測試結(jié)果:
我們可以在代碼示例中清晰的看到,在代理類中,代理對象包含了目標對象,并且在業(yè)務處理上進行了一定的業(yè)務擴展,但是卻和目標對象繼承于同一個接口。但是此擴展基于Spring AOP來講,以更加專業(yè)的叫法為前置增強、后置增強。
此類代理便是我們常說的靜態(tài)代理,靜態(tài)代理適合在業(yè)務比較簡單,實現(xiàn)類少,需求變化不頻繁,但是卻要對原有目標服務對象功能進行擴展,并且不去修改原有服務,這個時候我們就可以選擇使用靜態(tài)代理。
靜態(tài)代理的缺點
如果此時我們業(yè)務需要進行擴展,我們的代購同學在經(jīng)過市場調(diào)研以后,發(fā)現(xiàn)商品B更加受大家歡迎,這個時候我們就需要對自己的業(yè)務進行擴展了,怎么擴展呢?一起接著往下看。
抽象對象:
public interface ITargetFactoryBService { void saleB(String name); }
目標對象:
@Slf4j public class ITargetFactoryBServiceImpl implements ITargetFactoryBService { @Override public void saleB(String name) { log.info(name + "購買了商品B"); } }
代理對象:
@Slf4j public class ProxyTwoImpl implements ITargetFactoryService, ITargetFactoryBService { public ITargetFactoryService service; public ITargetFactoryBService bService; public ProxyTwoImpl(ITargetFactoryService service,ITargetFactoryBService bService){ super(); this.service = service; this.bService = bService; } @Override public void sale(String name) { before(); service.sale("代購"); after(); } @Override public void saleB(String name) { before(); bService.saleB("代購"); after(); } /** * 后置增強 */ private void after() { log.info("代購在購買后得到了市場調(diào)研結(jié)果"); } /** * 前置增強 */ private void before() { log.info("代購在購買前做了市場調(diào)研"); } }
測試類:
@Slf4j @SpringBootApplication(exclude = {DataSourceAutoConfiguration.class, DataSourceTransactionManagerAutoConfiguration.class, JdbcTemplateAutoConfiguration.class}) public class SpsringJdbcApplication { public static void main(String[] args) { TargetFactoryServiceImpl service = new TargetFactoryServiceImpl(); ITargetFactoryBServiceImpl bService = new ITargetFactoryBServiceImpl(); ProxyTwoImpl proxy2 = new ProxyTwoImpl(service, bService); proxy2.sale("代購"); proxy2.saleB("代購"); SpringApplication.run(SpsringJdbcApplication.class, args); } }
結(jié)果:
我們可以看到,在實現(xiàn)業(yè)務擴展的時候,需要對原有的代理類進行修改,如果后期我們需要擴展的業(yè)務較多的時候,這個類將變的更加繁雜,大量的繼承以及方法重寫,以至于牽一發(fā)而動全身,所以在這種業(yè)務擴展性高、業(yè)務變化頻繁的情況下我們不建議使用靜態(tài)代理。
靜態(tài)代理總結(jié):
1、違反Java設計模式開閉原則,即:程序?qū)ν鈹U展開放,對修改關閉。當需求進行變更時,我們應該是新增代碼塊來實現(xiàn),而不是在原來的代碼中進行修改實現(xiàn)。
2、擴展性很差。
3、可維護性差。
4、代碼耦合度高。
總結(jié)
到此這篇關于Java代理深入講解之靜態(tài)代理的文章就介紹到這了,更多相關Java靜態(tài)代理內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
java中List數(shù)組用逗號分隔開轉(zhuǎn)成字符串2種方法
在我們?nèi)粘i_發(fā)中,在前后端交互的時候會遇到多個id或其他字段存放到一個字段中,這時我們會遇到一個List(集合)---->String(單個字段),這篇文章主要給大家介紹了關于java中List數(shù)組用逗號分隔開轉(zhuǎn)成字符串的2種方法,需要的朋友可以參考下2023-10-10SpringMVC中的HandlerMapping和HandlerAdapter詳解
這篇文章主要介紹了SpringMVC中的HandlerMapping和HandlerAdapter詳解,在Spring MVC中,HandlerMapping(處理器映射器)用于確定請求處理器對象,請求處理器可以是任何對象,只要它們使用了@Controller注解或注解@RequestMapping,需要的朋友可以參考下2023-08-08關于pom.xml中maven無法下載springcloud包問題
小編遇到這樣一個問題spring-cloud-starter-feign,spring-cloud-starter-eureka 一直無法下載,maven倉庫中包路徑顯示為unknown,怎么解決呢?下面小編給大家?guī)砹藀om.xml中maven無法下載springcloud包問題,需要的朋友可以參考下2022-08-08spring boot如何使用AOP統(tǒng)一處理web請求
這篇文章主要介紹了spring boot如何使用AOP統(tǒng)一處理web請求,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2019-12-12Java中的Vector和ArrayList區(qū)別及比較
這篇文章主要介紹了Java中的Vector和ArrayList區(qū)別及比較,本文從API、同步、數(shù)據(jù)增長、使用模式4個方面總結(jié)了它們之間的不同之處,需要的朋友可以參考下2015-03-03如何在JDK 9中更簡潔使用 try-with-resources 語句
本文詳細介紹了自 JDK 7 引入的 try-with-resources 語句的原理和用法,以及介紹了 JDK 9 對 try-with-resources 的改進,使得用戶可以更加方便、簡潔的使用 try-with-resources 語句。,需要的朋友可以參考下2019-06-06java數(shù)據(jù)結(jié)構(gòu)排序算法之樹形選擇排序詳解
這篇文章主要介紹了java數(shù)據(jù)結(jié)構(gòu)排序算法之樹形選擇排序,結(jié)合具體實例形式分析了java樹形選擇排序的原理、實現(xiàn)技巧與相關注意事項,需要的朋友可以參考下2017-05-05