什么是代理?在我們生活中,特別是在朋友圈里,常??梢钥匆?jiàn)有些人代理賣產(chǎn)品,這個(gè)代理就相當(dāng)于交易的中間人。今天我們來(lái)聊聊Java中的代理模式,以及它的兩種模式動(dòng)態(tài)代理和靜態(tài)代理。
代理模式
代理模式(Proxy):為其他對(duì)象提供一個(gè)代理以控制對(duì)這個(gè)對(duì)象的訪問(wèn)。
主要解決:在直接訪問(wèn)對(duì)象時(shí)帶來(lái)的問(wèn)題,比如說(shuō):要訪問(wèn)的對(duì)象在遠(yuǎn)程的機(jī)器上。在面向?qū)ο笙到y(tǒng)中,有些對(duì)象由于某些原因(比如對(duì)象創(chuàng)建開(kāi)銷很大,或者某些操作需要安全控制,或者需要進(jìn)程外的訪問(wèn)),直接訪問(wèn)會(huì)給使用者或者系統(tǒng)結(jié)構(gòu)帶來(lái)很多麻煩,我們可以在訪問(wèn)此對(duì)象時(shí)加上一個(gè)對(duì)此對(duì)象的訪問(wèn)層。
代理模式的元素是:共同接口、代理對(duì)象、目標(biāo)對(duì)象。
代理模式的行為:由代理對(duì)象執(zhí)行目標(biāo)對(duì)象的方法、由代理對(duì)象擴(kuò)展目標(biāo)對(duì)象的方法。
代理模式的宏觀特性:對(duì)客戶端只暴露出接口,不暴露它以下的架構(gòu)。
好處多多:中間隔離了一層,更加符合開(kāi)閉原則
UML圖
創(chuàng)建一個(gè)接口
/**
* @Author: Promsing
* @Date: 2021/4/3 - 8:25
* @Description: 買車的接口
* @version: 1.0
*/
public interface BuyCar {
public void buyCar();
}
創(chuàng)建一個(gè)實(shí)現(xiàn)類
/**
* @Author: Promsing
* @Date: 2021/4/3 - 8:25
* @Description: 實(shí)現(xiàn)類
* @version: 1.0
*/
public class BuyCarImpl implements BuyCar {
@Override
public void buyCar() {
System.out.println("我要買車~~~啦啦啦");
}
}
靜態(tài)代理:
創(chuàng)建一個(gè)代理類
/**
* @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ù)注入,需要被代理的對(duì)象
public BuyCarProxy(final BuyCar buyCar) {
this.buyCar = buyCar;
}
//靜態(tài)代理- 的實(shí)現(xiàn)方式
@Override
public void buyCar() {
System.out.println("不貸款,全款!買車前的準(zhǔn)備~~~");
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)代理-+-+-+
不貸款,全款!買車前的準(zhǔn)備~~~
我要買車~~~啦啦啦
買完車了,出去浪~~~
動(dòng)態(tài)代理:
基于接口的動(dòng)態(tài)代理類
特點(diǎn):字節(jié)碼隨用隨創(chuàng)建,隨用隨加載
作用:在不修改源碼的基礎(chǔ)上對(duì)方法增強(qiáng)
涉及的類:JDK官方提供的Proxy
如何創(chuàng)建代理對(duì)象:使用Proxy類中的newProxyInstance方法
創(chuàng)建代理對(duì)象的要求:被代理類至少實(shí)現(xiàn)一個(gè)接口
newProxyInstance方法的參數(shù)
ClassLoader:類加載器,同于加載被代理對(duì)象字節(jié)碼
Class[]:字節(jié)碼數(shù)組---用于讓代理對(duì)象和被代理對(duì)象擁有相同的方法
InvocationHandler:用于提供被增強(qiáng)的代碼
/**
* @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 代理對(duì)象的引用
* @param method 當(dāng)前執(zhí)行的方法
* @param args 當(dāng)前執(zhí)行方法所需的參數(shù)
* @return 和被代理對(duì)象方法有相同的返回值
* @throws Throwable
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("不貸款,全款!買車前的準(zhǔn)備~~~");
Object result = method.invoke(object, args);
System.out.println("買完車了,出去浪~~~");
return result;
}
}
客戶端
public static void main(String[] args) {
System.out.println("-+-+-+使用基于接口的代理-+-+-+");
//方式一、如不寫動(dòng)態(tài)代理類DynamicProxy,可以在這里使用內(nèi)部類
//聲明一個(gè)final修飾的對(duì)象
/*
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("不貸款,全款!買車前的準(zhǔn)備~~~");
Object result = method.invoke(car, args);
System.out.println("買完車了,出去浪~~~");
return result;
}
});
proxy.buyCar();
*/
//方式二、使用DynamicProxy類
//聲明一個(gè)final修飾的對(duì)象
final BuyCarImpl car=new BuyCarImpl();
BuyCar proxy=(BuyCar)Proxy.newProxyInstance(car.getClass().getClassLoader(), car.getClass().getInterfaces(),new DynamicProxy(car));
proxy.buyCar();
}
基于子類的動(dòng)態(tài)代理
特點(diǎn):字節(jié)碼隨用隨創(chuàng)建,隨用隨加載
作用:在不修改源碼的基礎(chǔ)上對(duì)方法增強(qiáng)
涉及的類:第三方cglib提供的Enhancer
如何創(chuàng)建代理對(duì)象:使用Enhancer類中create方法
創(chuàng)建代理對(duì)象的要求:被代理類不能是最終類
newProxyInstance方法的參數(shù)
Class:用于被指定代理對(duì)象的字節(jié)碼
InvocationHandler:用于提供增強(qiáng)的方法
public static void main(String[] args) {
//使用基于子類的動(dòng)態(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("不貸款,全款!買車前的準(zhǔn)備~~~");
Object result = method.invoke(car, args);
System.out.println("買完車了,出去浪~~~");
return result;
}
});
proxy.buyCar();
}
總結(jié)
看到這里小伙伴,對(duì)于Java中的動(dòng)態(tài)代理和靜態(tài)代理有一個(gè)概念,想要了解更多相關(guān)Java靜態(tài)代理和動(dòng)態(tài)代理內(nèi)容請(qǐng)搜索W3Cschool以前的文章或繼續(xù)瀏覽下面的相關(guān)文章!