在代理模式(Proxy Pattern)中,一個(gè)類代表另一個(gè)類的功能。這種類型的設(shè)計(jì)模式屬于結(jié)構(gòu)型模式。
在代理模式中,我們創(chuàng)建具有現(xiàn)有對(duì)象的對(duì)象,以便向外界提供功能接口。
意圖:為其他對(duì)象提供一種代理以控制對(duì)這個(gè)對(duì)象的訪問(wèn)。
JDK 自帶的動(dòng)態(tài)代理
- java.lang.reflect.Proxy:生成動(dòng)態(tài)代理類和對(duì)象;
- java.lang.reflect.InvocationHandler(處理器接口):可以通過(guò)invoke方法實(shí)現(xiàn)對(duì)真實(shí)角色的代理訪問(wèn)。
- 每次通過(guò) Proxy 生成的代理類對(duì)象都要指定對(duì)應(yīng)的處理器對(duì)象。
實(shí)現(xiàn)
? 我們將創(chuàng)建一個(gè) BaseService接口和實(shí)現(xiàn)了 BaseService接口的實(shí)體類。Invocation是一個(gè)處理器類,通過(guò)Invocation來(lái)的invoke方法來(lái)執(zhí)行或補(bǔ)充真實(shí)類的需求。ProxyFactory是一個(gè)代理類工廠,通過(guò)Proxy類來(lái)生成動(dòng)態(tài)代理類和對(duì)象。
步驟一
創(chuàng)建一個(gè)接口,實(shí)現(xiàn)這個(gè)接口的類才能被動(dòng)態(tài)代理
/**
* 只有被需要被監(jiān)控的行為才有資格在這里聲明
*/
public interface BaseService {
void eat();
void wc();
}
步驟二
創(chuàng)建實(shí)現(xiàn)接口的實(shí)體類。
public class Person implements BaseService {
@Override
public void eat() { //主要業(yè)務(wù),代理模式要求開發(fā)人員只關(guān)心主要業(yè)務(wù)
System.out.println("使用筷子吃飯。。。");
}
@Override
public void wc() {
System.out.println("測(cè)試重力是否存在。。。");
}
}
步驟三
創(chuàng)建實(shí)現(xiàn)了InvocationHandler 接口的處理器類
public class Invocation implements InvocationHandler {
private BaseService obj; //具體被監(jiān)控對(duì)象
public Invocation(BaseService param){
obj=param;
}
/**
* 在被監(jiān)控的行為將要執(zhí)行時(shí),會(huì)被JVM攔截。被監(jiān)控行為(方法)和行為實(shí)現(xiàn)方(類實(shí)例)會(huì)作為參數(shù)輸送invoke
* 通知JVM,這個(gè)被攔截方法是如何與當(dāng)前次要業(yè)務(wù)方法綁定實(shí)現(xiàn)
*
* @param proxy 將輔助監(jiān)控行為實(shí)現(xiàn)方(類實(shí)例)的代理對(duì)象作為proxy
* @param method 被監(jiān)控行為(方法)封裝成Method類型對(duì)象
* @param args 被監(jiān)控行為(方法)運(yùn)行時(shí)接收所有的實(shí)參封裝到Object[]
* @return
* @throws Throwable
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//0.局部變量,接受主要業(yè)務(wù)方法執(zhí)行完畢后的返回值
Object value;
//1.確認(rèn)被攔截的行為
String methodName = method.getName();
//2.根據(jù)別攔截的行為不同,決定主要業(yè)務(wù)與次要業(yè)務(wù)如何綁定執(zhí)行
if ("eat".equals(methodName)) { //飯前要洗手
wash(); //洗手
value = method.invoke(this.obj, args);//吃飯
}else{
value = method.invoke(this.obj,args); //廁所
wash(); //洗手
}
return value; //返回到被攔截方法需要調(diào)用的地方
}
//次要業(yè)務(wù)
public void wash(){
System.out.println("--------洗手---------");
}
}
步驟四
創(chuàng)建一個(gè)代理類工廠,通過(guò)Proxy類來(lái)生成動(dòng)態(tài)代理類和對(duì)象。
public class ProxyFactory {
/**
* JDK動(dòng)態(tài)代理模式下,代理對(duì)象的數(shù)據(jù)類型,應(yīng)該由監(jiān)控行為來(lái)描述
* 參數(shù):Class文件,監(jiān)控類。
*/
public static BaseService Builder(Class clazz) throws IllegalAccessException, InstantiationException {
//1.創(chuàng)建被監(jiān)控實(shí)例對(duì)象
BaseService obj = (BaseService) clazz.newInstance();
//2.創(chuàng)建一個(gè)通知(處理器)對(duì)象
InvocationHandler adviser = new Invocation(obj);
//3.想JVM申請(qǐng)負(fù)責(zé)監(jiān)控obj對(duì)象指定行為的監(jiān)控對(duì)象(代理對(duì)象)
/**
* loader:被監(jiān)控對(duì)象隸屬的類文件在我們內(nèi)存中真實(shí)地址
* interfaces:被監(jiān)控對(duì)象隸屬的類文件實(shí)現(xiàn)接口
* h:監(jiān)控對(duì)象發(fā)現(xiàn)類實(shí)例要執(zhí)行被監(jiān)控行為,應(yīng)該由哪一個(gè)通知對(duì)象進(jìn)行輔助
*/
BaseService $proxy = (BaseService)
Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), adviser);
return $proxy;
}
}
步驟五
測(cè)試類
public class TestMain {
public static void main(String[] args) throws InstantiationException, IllegalAccessException {
//Person mike = new Person();
BaseService mike=ProxyFactory.Builder(Person.class); //返回的是一個(gè)代理對(duì)象 不是mike
mike.eat();
mike.wc();
}
}

測(cè)試結(jié)果.png