Java結(jié)構(gòu)型設(shè)計模式中代理模式示例詳解
代理模式
代理模式(Proxy Pattern)屬于結(jié)構(gòu)型模式。
它是指為其他對象提供一種代理以控制對這個對象的訪問。
在某些情況下,一個對象不適合或者不能直接引用另一個對象,而代理對象可以在客戶端和目標(biāo)對象之間起到中介的作用。
在代理模式中,創(chuàng)建具有現(xiàn)有對象的對象,以便向外界提供功能接口。
想在訪問一個類時做一些控制的時候就可以是用代理模式。
分類
代理模式屬于結(jié)構(gòu)型模式,分為靜態(tài)代理和動態(tài)代理。
1.靜態(tài)代理:靜態(tài)定義代理類
靜態(tài)代理需要自己生成代理類
2.動態(tài)代理:動態(tài)生成代理類
動態(tài)代理不用親自去實現(xiàn),通常使用現(xiàn)成的API即可。目前普遍使用的是JDK自帶的代理與CGLIB提供的類庫。
主要角色
代理模式一般包含三種角色:
1.抽象主題角色(Subject)
抽象主題類的主要職責(zé)是聲明真實主題與代理的共同接口方法,該類可以是接口也可以是抽象類
2.真實主題角色(RealSubject)
該類也被稱為被代理類,該類定義了代理所表示的真實對象,是負(fù)責(zé)執(zhí)行系統(tǒng)真正的羅輯業(yè)務(wù)對象
3.代理主題角色(Proxy)
代理主題也被稱為代理類,其內(nèi)部特有RealSubject的引用,因此具備完全的對RealSubject的代理權(quán)。
客戶端調(diào)用代理對象的方法,同時也調(diào)用被代理對象的方法,但是會在代理對象前后增加一些處理代碼。可以理解為代碼增強,實際上就是在原代碼羅輯前后增加一些代碼邏輯,而使調(diào)用者無感知。
作用
1.保護(hù)目標(biāo)對象,將代理對象與真實被調(diào)用目標(biāo)對象分離
2.增強目標(biāo)對象
3.降低系統(tǒng)耦合性,提升擴展性
靜態(tài)代理與動態(tài)代理的區(qū)別
1.靜態(tài)代理只能通過手動完成代理操作,如果被代理類增加了新的方法,代理類需要同步增加,違背開閉原則。
2.動態(tài)代理采用在運行時動態(tài)生成代碼的方式,取消了對被代理類的擴展限制,遵循開閉原測。
3.若動態(tài)代理要對目標(biāo)類的增強邏輯進(jìn)行擴展,結(jié)合策略模式,只需要新增策略類便可完成,無須修改代理類的代碼。
靜態(tài)代理的基本使用
靜態(tài)代理需要自己生成代理類
創(chuàng)建抽象主題
public interface ISubject { /** * 買票 */ void buyTickets(); }
創(chuàng)建真實主題
public class RealSubject implements ISubject { public void buyTickets() { System.out.println("進(jìn)行買票操作"); } }
創(chuàng)建代理主題
public class Proxy implements ISubject { private ISubject subject; public Proxy(ISubject subject) { this.subject = subject; } public void buyTickets() { before(); subject.buyTickets(); after(); } public void before() { System.out.println("買票前的操作"); } public void after() { System.out.println("買票后的操作"); } }
客戶端調(diào)用
public static void main(String[] args) { Proxy proxy = new Proxy(new RealSubject()); proxy.buyTickets(); }
買票前的操作
進(jìn)行買票操作
買票后的操作
JDK動態(tài)代理的基本使用
創(chuàng)建抽象主題
public interface IUser { /** * 購物 */ void shopping(); }
創(chuàng)建真實主題
public class User implements IUser{ @Override public void shopping() { System.out.println("user shopping...."); } }
創(chuàng)建代理主題
public class JDKProxy implements InvocationHandler { private Object tarjet; public JDKProxy(Object tarjet) { this.tarjet = tarjet; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("代理user,執(zhí)行shopping()開始..."); Object oj = method.invoke(tarjet, args); System.out.println("代理user,執(zhí)行shopping()結(jié)束..."); return oj; } }
客戶端調(diào)用
public static void main(String[] args) { User user = new User(); JDKProxy jdkProxy = new JDKProxy(user); IUser proxyInstance = (IUser) Proxy.newProxyInstance(user.getClass().getClassLoader(), user.getClass().getInterfaces(), jdkProxy); proxyInstance.shopping(); }
代理user,執(zhí)行shopping()開始...
user shopping....
代理user,執(zhí)行shopping()結(jié)束...
小優(yōu)化
在調(diào)用時候,傳入了一推參數(shù),可進(jìn)一步優(yōu)化
public class JDKProxy implements InvocationHandler { private Object tarjet; public Object JDKProxy(Object target){ this.tarjet = target; Class<?> clazz = target.getClass(); return Proxy.newProxyInstance(clazz.getClassLoader(),clazz.getInterfaces(),this); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("代理user,執(zhí)行shopping()開始..."); Object oj = method.invoke(tarjet, args); System.out.println("代理user,執(zhí)行shopping()結(jié)束..."); return oj; } }
public static void main(String[] args) { JDKProxy jdkProxy = new JDKProxy(); IUser user= (IUser)jdkProxy.JDKProxy(new User()); user.shopping(); }
CGLIB動態(tài)代理的基本使用
CGLIB動態(tài)代理也不需要生成代理類,實現(xiàn)MethodInterceptor 就可以了。
注意:CGLib不能代理final的方法
創(chuàng)建抽象主題
注意:CGLb代理的目標(biāo)對象不需要實現(xiàn)任何接口,就可以通過動態(tài)繼承目標(biāo)對象實現(xiàn)動態(tài)代理。所以此處可以不用創(chuàng)建接口。直接使用真實主題。
public interface IUser { public void shopping(); }
創(chuàng)建真實主題
public class User implements IUser{ @Override public void shopping() { System.out.println("user shopping...."); } }
直接使用真實主題。
public class User { public void shopping() { System.out.println("user shopping...."); } }
創(chuàng)建代理主題
public class CglibProxy implements MethodInterceptor { @Override public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { System.out.println("代理user,執(zhí)行shopping()開始..."); Object invokeSuper = methodProxy.invokeSuper(o, objects); System.out.println("代理user,執(zhí)行shopping()結(jié)束..."); return invokeSuper; } }
客戶端調(diào)用
public static void main(String[] args) { CglibProxy cglibProxy = new CglibProxy(); Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(User.class); enhancer.setCallback(cglibProxy); IUser iUser = (IUser) enhancer.create(); iUser.shopping(); }
小優(yōu)化
在客戶端調(diào)用時,稍顯復(fù)雜,可進(jìn)一步優(yōu)化
public class CglibProxy implements MethodInterceptor { public Object getInstance(Class<?> clazz) throws Exception{ //相當(dāng)于Proxy,代理工具類 Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(clazz); enhancer.setCallback(this); return enhancer.create(); } @Override public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { System.out.println("代理user,執(zhí)行shopping()開始..."); Object invokeSuper = methodProxy.invokeSuper(o, objects); System.out.println("代理user,執(zhí)行shopping()結(jié)束..."); return invokeSuper; } }
public static void main(String[] args) throws Exception { IUser user = (IUser) new CglibProxy().getInstance(User.class); user.shopping(); }
CGLIB與JDK動態(tài)代理區(qū)別
1.執(zhí)行條件
JDK動態(tài)代理實現(xiàn)了被代理對象的接口。CGLb代理的目標(biāo)對象不需要實現(xiàn)任何接口,它是通過動態(tài)繼承目標(biāo)對象實現(xiàn)動態(tài)代理。
2.實現(xiàn)機制
JDK動態(tài)代理調(diào)用代理方法是由Java內(nèi)部的反射機制來實現(xiàn)的,需要讀取接口信息。CGLib動態(tài)代理是通過FastClass機制來實現(xiàn)的,需要覆蓋父類方法。
3.性能
首先都在運行期生成字節(jié)碼。
JDK動態(tài)代理的代理邏輯簡單,直接寫Class字節(jié)碼,使用反射機制在生成類的過程中比較高效。
CGLib代理實現(xiàn)更復(fù)雜,使用ASM框架寫Class字節(jié)碼,但是asm在生成類之后的相關(guān)執(zhí)行過程中比較高效。但是可以通過將asm生成的類進(jìn)行緩存,解決asm生成類過程低效問題。
一句話:CGLib生成代理類比JDK動態(tài)代理效率低,但是執(zhí)行效率比JDK動態(tài)代理高。
到此這篇關(guān)于Java結(jié)構(gòu)型設(shè)計模式中代理模式示例詳解的文章就介紹到這了,更多相關(guān)Java代理模式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java操作mongodb增刪改查的基本操作實戰(zhàn)指南
MongoDB是一個基于分布式文件存儲的數(shù)據(jù)庫,由c++語言編寫,旨在為WEB應(yīng)用提供可擴展的高性能數(shù)據(jù)存儲解決方案,下面這篇文章主要給大家介紹了關(guān)于Java操作mongodb增刪改查的基本操作實戰(zhàn)指南,需要的朋友可以參考下2023-05-05spring @schedule注解如何動態(tài)配置時間間隔
這篇文章主要介紹了spring @schedule注解如何動態(tài)配置時間間隔,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-11-11SpringBoot使用Druid數(shù)據(jù)源的配置方法
這篇文章主要介紹了SpringBoot使用Druid數(shù)據(jù)源的配置方法,文中代碼實例相結(jié)合的形式給大家介紹的非常詳細(xì),需要的朋友參考下吧2018-04-04如何實現(xiàn)nohup?java進(jìn)程號一直在變方法步驟詳解
這篇文章主要為大家介紹了如何實現(xiàn)nohup?java進(jìn)程號一直在變方法步驟詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11Java中Object轉(zhuǎn)換為List類型的實現(xiàn)方法
這篇文章主要介紹了Java中Object轉(zhuǎn)換為List類型的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03