Java的設(shè)計(jì)模式之代理模式使用詳解
一、代理模式介紹
代理模式(Proxy Pattern),是23種設(shè)計(jì)模式之一,它關(guān)心的主要是過(guò)程,而不是結(jié)果。
代理模式主要提供了對(duì)目標(biāo)對(duì)象的間接訪問(wèn)方式,即通過(guò)代理對(duì)象來(lái)訪問(wèn)目標(biāo)對(duì)象,這樣可以在目標(biāo)對(duì)象實(shí)現(xiàn)的基礎(chǔ)上,增強(qiáng)額外的功能操作,即擴(kuò)展目標(biāo)對(duì)象的功能,SpringAop便是一個(gè)很好的例子。
而被代理的對(duì)象可以是遠(yuǎn)程對(duì)象、創(chuàng)建開(kāi)銷大的對(duì)象或需要安全控制的對(duì)象。
代理模式的簡(jiǎn)單示意圖如下。
代理模式主要有三種實(shí)現(xiàn)方式 靜態(tài)代理、動(dòng)態(tài)代理(JDK代理、接口代理)和Cglib代理(不需要實(shí)現(xiàn)接口)。
二、三種代理模式的簡(jiǎn)單介紹
1、靜態(tài)代理
介紹
靜態(tài)代理UML類圖如下圖,StudentDaoProxy類是代理類,通過(guò)聚合的方式代理StudentDao類,同時(shí)StudentDao類和代理類都實(shí)現(xiàn)IStudentDao接口類。
靜態(tài)代理的優(yōu)點(diǎn)是能夠較為簡(jiǎn)單快速的在不修改目標(biāo)對(duì)象的前提下,對(duì)目標(biāo)對(duì)象功能進(jìn)行擴(kuò)展;而缺點(diǎn)就是不夠靈活,如果接口新增方法,那么需要維護(hù)類的成本太大。
代碼實(shí)現(xiàn)
①新建IStudentDao接口類
public interface IStudentDao { void read(); }
②被代理的學(xué)生類
public class StudentDao implements IStudentDao{ @Override public void read() { System.out.println("正在讀書(shū)中。。。。。"); } }
③創(chuàng)建代理類
public class StudentDaoProxy implements IStudentDao{ // 被代理類聚合到代理類 private StudentDao studentDao; public StudentDaoProxy(StudentDao studentDao) { // 通過(guò)構(gòu)造器傳入聚合對(duì)象,也可以通過(guò)set方法傳入 this.studentDao=studentDao; } @Override public void read() { // 增強(qiáng)方法,這里可以寫(xiě)其他復(fù)雜業(yè)務(wù) System.out.println("打開(kāi)書(shū)本。。。。。"); //被代理類原方法 studentDao.read(); } }
④用戶測(cè)試類
public class Client { public static void main(String[] args) { StudentDao studentDao = new StudentDao(); StudentDaoProxy studentDaoProxy = new StudentDaoProxy(studentDao); // 使用代理方法 studentDaoProxy.read(); } }
輸出結(jié)果
打開(kāi)書(shū)本。。。。。
正在讀書(shū)中。。。。。
2、動(dòng)態(tài)代理
介紹
動(dòng)態(tài)代理對(duì)象不需要實(shí)現(xiàn)接口,但是目標(biāo)對(duì)象還是需要實(shí)現(xiàn)接口。
而代理對(duì)象的生成,是利用JDK的API,動(dòng)態(tài)的在內(nèi)存中構(gòu)建代理對(duì)象,相比于靜態(tài)代理靈活的多,所以也叫JDK代理或接口代理。
其UML類圖如下
代碼實(shí)現(xiàn)
①新建IStudentDao接口類
public interface IStudentDao { void read(); }
②被代理的學(xué)生類
public class StudentDao implements IStudentDao{ @Override public void read() { System.out.println("正在讀書(shū)中。。。。。"); } }
③創(chuàng)建代理類
這里相比靜態(tài)代理有點(diǎn)小修改,JDK代理需要使用newProxyInstance方法,其完整寫(xiě)法是static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandlerh),該方法在java.lang.reflect.Proxy包內(nèi),通過(guò)反射機(jī)制實(shí)現(xiàn)。
public class ProxyFactory{ // 被代理類聚合到代理類 private Object target; public ProxyFactory(Object studentDao) { // 通過(guò)構(gòu)造器傳入聚合對(duì)象,也可以通過(guò)set方法傳入 this.target=studentDao; } //給目標(biāo)對(duì)象 生成一個(gè)代理對(duì)象 public Object getProxyInstance() { //說(shuō)明 /* * public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) //1. ClassLoader loader : 指定當(dāng)前目標(biāo)對(duì)象使用的類加載器, 獲取加載器的方法固定 //2. Class<?>[] interfaces: 目標(biāo)對(duì)象實(shí)現(xiàn)的接口類型,使用泛型方法確認(rèn)類型 //3. InvocationHandler h : 事情處理,執(zhí)行目標(biāo)對(duì)象的方法時(shí),會(huì)觸發(fā)事情處理器方法, 會(huì)把當(dāng)前執(zhí)行的目標(biāo)對(duì)象方法作為參數(shù)傳入 */ return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), new MyInvocationHandler()); } class MyInvocationHandler implements InvocationHandler{ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { // 自定義業(yè)務(wù)邏輯 System.out.println("JDK代理開(kāi)始~~"); //反射機(jī)制調(diào)用目標(biāo)對(duì)象的方法 Object returnVal = method.invoke(target, args); System.out.println("JDK代理提交"); return returnVal; } } }
④用戶測(cè)試類
public class Client { public static void main(String[] args) { StudentDao studentDao = new StudentDao(); ProxyFactory studentDaoProxy = new ProxyFactory(studentDao); // 使用代理方法,注意這里一定要用接口接收 IStudentDao studentDao2 = (IStudentDao) studentDaoProxy.getProxyInstance(); studentDao2.read(); } }
結(jié)果
JDK代理開(kāi)始~~
正在讀書(shū)中。。。。。
JDK代理提交
3、Cglib代理
介紹
- 動(dòng)態(tài)代理或靜態(tài)代理都要求實(shí)現(xiàn)一個(gè)接口,但是對(duì)于Cglib并不需要實(shí)現(xiàn)任何接口。
- Cglib代理也叫作子類代理,它是在內(nèi)存中構(gòu)建一個(gè)子類對(duì)象從而實(shí)現(xiàn)對(duì)目標(biāo)對(duì)象功能擴(kuò)展,所以也可稱為動(dòng)態(tài)代理。
- Cglib是一個(gè)強(qiáng)大的高性能的代碼生成包,它可以在運(yùn)行期擴(kuò)展java類與實(shí)現(xiàn)java接口。
- 它廣泛的被許多AOP框架使用,例如SpringAOP,實(shí)現(xiàn)方法攔截。
- 在AOP編程中如何選擇代理模式:
- 目標(biāo)對(duì)象需要實(shí)現(xiàn)接口,用JDK代理
- 目標(biāo)對(duì)象不需要實(shí)現(xiàn)接口,用Cglib代理
- Cglib包的底層是通過(guò)使用字節(jié)碼處理框架ASM來(lái)轉(zhuǎn)換字節(jié)碼并生成新的類
代碼實(shí)現(xiàn)
使用Cglib時(shí)代理的類不能為final/static,否則不會(huì)被攔截,同時(shí)需要引入cglib相關(guān)jar包
①被代理的學(xué)生類
public class StudentDao implements IStudentDao{ @Override public void read() { System.out.println("正在讀書(shū)中。。。。。"); } }
②創(chuàng)建代理類
public class ProxyFactory implements MethodInterceptor{ //維護(hù)一個(gè)目標(biāo)對(duì)象 private Object target; //構(gòu)造器,傳入一個(gè)被代理的對(duì)象 public ProxyFactory(Object target) { this.target = target; } //返回一個(gè)代理對(duì)象: 是 target 對(duì)象的代理對(duì)象 public Object getProxyInstance() { //1. 創(chuàng)建一個(gè)工具類 Enhancer enhancer = new Enhancer(); //2. 設(shè)置父類 enhancer.setSuperclass(target.getClass()); //3. 設(shè)置回調(diào)函數(shù) enhancer.setCallback(this); //4. 創(chuàng)建子類對(duì)象,即代理對(duì)象 return enhancer.create(); } //重寫(xiě) intercept 方法,會(huì)調(diào)用目標(biāo)對(duì)象的方法 @Override public Object intercept(Object arg0, Method method, Object[] args, MethodProxy arg3) throws Throwable { // TODO Auto-generated method stub System.out.println("Cglib代理模式 ~~ 開(kāi)始"); Object returnVal = method.invoke(target, args); System.out.println("Cglib代理模式 ~~ 提交"); return returnVal; } }
③用戶測(cè)試類
public class Client { public static void main(String[] args) { StudentDao studentDao = new StudentDao(); ProxyFactory studentDaoProxy = new ProxyFactory(studentDao); StudentDao studentDao2 = (StudentDao)studentDaoProxy.getProxyInstance(); studentDao2.read(); } }
結(jié)果
Cglib代理模式 ~~ 開(kāi)始
正在讀書(shū)中。。。。。
Cglib代理模式 ~~ 提交
到此這篇關(guān)于Java的設(shè)計(jì)模式之代理模式使用詳解的文章就介紹到這了,更多相關(guān)Java代理模式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
關(guān)于java中構(gòu)造函數(shù)的一些知識(shí)詳解
下面小編就為大家?guī)?lái)一篇關(guān)于java中構(gòu)造函數(shù)的一些知識(shí)詳解。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-12-12springSecurity實(shí)現(xiàn)簡(jiǎn)單的登錄功能
這篇文章主要為大家詳細(xì)介紹了springSecurity實(shí)現(xiàn)簡(jiǎn)單的登錄功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-09-09java設(shè)計(jì)模式之實(shí)現(xiàn)對(duì)象池模式示例分享
對(duì)象池模式經(jīng)常用在頻繁創(chuàng)建、銷毀對(duì)象(并且對(duì)象創(chuàng)建、銷毀開(kāi)銷很大)的場(chǎng)景,比如數(shù)據(jù)庫(kù)連接池、線程池、任務(wù)隊(duì)列池等。本代碼簡(jiǎn)單,沒(méi)有限制對(duì)象池大小2014-02-02Java?Socket編程從零到實(shí)戰(zhàn)詳解(完整實(shí)戰(zhàn)案例)
這篇文章主要介紹了Java?Socket編程從零到實(shí)戰(zhàn)詳解,本文給大家介紹的非常詳細(xì),感興趣的朋友一起看看吧2025-04-04Spring Boot 之HelloWorld開(kāi)發(fā)案例
這篇文章主要介紹了Spring Boot 之HelloWorld開(kāi)發(fā)案例,需要的朋友可以參考下2017-04-04