Java靜態(tài)代理與動態(tài)代理案例詳解
代理模式
代理模式(Proxy):為其他對象提供一個代理以控制對這個對象的訪問。
主要解決:在直接訪問對象時帶來的問題,比如說:要訪問的對象在遠程的機器上。在面向?qū)ο笙到y(tǒng)中,有些對象由于某些原因(比如對象創(chuàng)建開銷很大,或者某些操作需要安全控制,或者需要進程外的訪問),直接訪問會給使用者或者系統(tǒng)結(jié)構(gòu)帶來很多麻煩,我們可以在訪問此對象時加上一個對此對象的訪問層。
代理模式的元素是:共同接口、代理對象、目標對象。
代理模式的行為:由代理對象執(zhí)行目標對象的方法、由代理對象擴展目標對象的方法。
代理模式的宏觀特性:對客戶端只暴露出接口,不暴露它以下的架構(gòu)。
好處多多:中間隔離了一層,更加符合開閉原則
UML圖

創(chuàng)建一個接口
/**
* @Author: Promsing
* @Date: 2021/4/3 - 8:25
* @Description: 買車的接口
* @version: 1.0
*/
public interface BuyCar {
public void buyCar();
}
創(chuàng)建一個實現(xiàn)類
/**
* @Author: Promsing
* @Date: 2021/4/3 - 8:25
* @Description: 實現(xiàn)類
* @version: 1.0
*/
public class BuyCarImpl implements BuyCar {
@Override
public void buyCar() {
System.out.println("我要買車~~~啦啦啦");
}
}
靜態(tài)代理:
創(chuàng)建一個代理類
/**
* @Author: Promsing
* @Date: 2021/4/3 - 8:26
* @Description: 代理類
* @version: 1.0
*/
public class BuyCarProxy implements BuyCar{
private BuyCar buyCar;
//注意事final修飾的關(guān)鍵字 不可修改
//構(gòu)造函數(shù)注入,需要被代理的對象
public BuyCarProxy(final BuyCar buyCar) {
this.buyCar = buyCar;
}
//靜態(tài)代理- 的實現(xiàn)方式
@Override
public void buyCar() {
System.out.println("不貸款,全款!買車前的準備~~~");
buyCar.buyCar();
System.out.println("買完車了,出去浪~~~");
}
}
客戶端調(diào)用
/**
* @Author: Promsing
* @Date: 2021/4/3 - 8:36
* @Description: 客戶端調(diào)用
* @version: 1.0
*/
public abstract class ProxyTest implements BuyCar {
public static void main(String[] args) {
System.out.println("-+-+-+正常調(diào)用-+-+-+");
BuyCar car=new BuyCarImpl();
car.buyCar();
System.out.println("-+-+-+使用靜態(tài)代理-+-+-+");
BuyCar proxy=new BuyCarProxy(car);
proxy.buyCar();
}
}
-+-+-+正常調(diào)用-+-+-+
我要買車~~~啦啦啦
-+-+-+使用靜態(tài)代理-+-+-+
不貸款,全款!買車前的準備~~~
我要買車~~~啦啦啦
買完車了,出去浪~~~
動態(tài)代理:
基于接口的動態(tài)代理類
特點:字節(jié)碼隨用隨創(chuàng)建,隨用隨加載
作用:在不修改源碼的基礎(chǔ)上對方法增強
涉及的類:JDK官方提供的Proxy
如何創(chuàng)建代理對象:使用Proxy類中的newProxyInstance方法
創(chuàng)建代理對象的要求:被代理類至少實現(xiàn)一個接口
newProxyInstance方法的參數(shù)
ClassLoader:類加載器,同于加載被代理對象字節(jié)碼
Class[]:字節(jié)碼數(shù)組---用于讓代理對象和被代理對象擁有相同的方法
InvocationHandler:用于提供被增強的代碼
/**
* @Author: Promsing
* @Date: 2021/4/3 - 9:09
* @Description: 描述 形容
* @version: 1.0
*/
public class DynamicProxy implements InvocationHandler {
private BuyCar object;
public DynamicProxy( BuyCar object) {
this.object = object;
}
/**
*
* @param proxy 代理對象的引用
* @param method 當前執(zhí)行的方法
* @param args 當前執(zhí)行方法所需的參數(shù)
* @return 和被代理對象方法有相同的返回值
* @throws Throwable
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("不貸款,全款!買車前的準備~~~");
Object result = method.invoke(object, args);
System.out.println("買完車了,出去浪~~~");
return result;
}
}
客戶端
public static void main(String[] args) {
System.out.println("-+-+-+使用基于接口的代理-+-+-+");
//方式一、如不寫動態(tài)代理類DynamicProxy,可以在這里使用內(nèi)部類
//聲明一個final修飾的對象
/*
final BuyCarImpl car=new BuyCarImpl();
BuyCar proxy=(BuyCar)Proxy.newProxyInstance(car.getClass().getClassLoader(), car.getClass().getInterfaces(), new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("不貸款,全款!買車前的準備~~~");
Object result = method.invoke(car, args);
System.out.println("買完車了,出去浪~~~");
return result;
}
});
proxy.buyCar();
*/
//方式二、使用DynamicProxy類
//聲明一個final修飾的對象
final BuyCarImpl car=new BuyCarImpl();
BuyCar proxy=(BuyCar)Proxy.newProxyInstance(car.getClass().getClassLoader(), car.getClass().getInterfaces(),new DynamicProxy(car));
proxy.buyCar();
}
基于子類的動態(tài)代理
特點:字節(jié)碼隨用隨創(chuàng)建,隨用隨加載
作用:在不修改源碼的基礎(chǔ)上對方法增強
涉及的類:第三方cglib提供的Enhancer
如何創(chuàng)建代理對象:使用Enhancer類中create方法
創(chuàng)建代理對象的要求:被代理類不能是最終類
newProxyInstance方法的參數(shù)
Class:用于被指定代理對象的字節(jié)碼
InvocationHandler:用于提供增強的方法
public static void main(String[] args) {
//使用基于子類的動態(tài)代理
//需要引入Jar包--cglib 本案例使用cglib3.3.0
System.out.println("-+-+-+使用基于子類的代理-+-+-+");
final BuyCarImpl car=new BuyCarImpl();
BuyCar proxy= (BuyCar)Enhancer.create(car.getClass(), new MethodInterceptor() {
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("不貸款,全款!買車前的準備~~~");
Object result = method.invoke(car, args);
System.out.println("買完車了,出去浪~~~");
return result;
}
});
proxy.buyCar();
}
到此這篇關(guān)于Java靜態(tài)代理與動態(tài)代理案例詳解的文章就介紹到這了,更多相關(guān)Java靜態(tài)代理與動態(tài)代理內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
IntelliJ?IDEA?2023.2最新版激活方法及驗證ja-netfilter配置是否成功
隨著2023.2版本的發(fā)布,用戶們渴望了解如何激活這個最新版的IDE,本文將介紹三種可行的激活方案,包括許可證服務(wù)器、許可證代碼和idea?vmoptions配置,幫助讀者成功激活并充分利用IDEA的功能,感興趣的朋友參考下吧2023-08-08
SpringBoot參數(shù)校驗的一些實戰(zhàn)應(yīng)用
這篇文章主要給大家介紹了關(guān)于SpringBoot參數(shù)校驗的一些實戰(zhàn)應(yīng)用,包括使用內(nèi)置的參數(shù)校驗注解、嵌套對象校驗、分組校驗以及自定義校驗注解,通過這些方法,可以有效地提高系統(tǒng)的穩(wěn)定性和安全性,需要的朋友可以參考下2024-11-11
Java利用數(shù)組隨機抽取幸運觀眾如何實現(xiàn)
這篇文章主要介紹了Java利用數(shù)組隨機抽取幸運觀眾如何實現(xiàn),需要的朋友可以參考下2014-02-02
SpringCloud組件之Eureka Server詳細啟動過程及說明
這篇文章主要介紹了SpringCloud組件之Eureka Server詳細啟動過程及說明,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01
Java中的ConcurrentLinkedQueue使用解析
這篇文章主要介紹了Java中的ConcurrentLinkedQueue使用解析,一個基于鏈接節(jié)點的無界線程安全隊列,此隊列按照 FIFO(先進先出)原則對元素進行排序,隊列的頭部是隊列中時間最長的元素,需要的朋友可以參考下2023-12-12
Java注解的Retention和RetentionPolicy實例分析
這篇文章主要介紹了Java注解的Retention和RetentionPolicy,結(jié)合實例形式分析了Java注解Retention和RetentionPolicy的基本功能及使用方法,需要的朋友可以參考下2019-09-09
Java入門基礎(chǔ)之常規(guī)的命名方法和變量的值及其引用
這篇文章主要介紹了Java的命名方法和變量的值及其引用,是Java入門學習中的基礎(chǔ)知識,需要的朋友可以參考下2015-09-09

