代理模式箩溃,顧名思義,就是找個(gè)代理人碌嘀。就好像房產(chǎn)中介,當(dāng)你看上一個(gè)房子想買的時(shí)候歪架,并不是直接跟賣主聯(lián)系股冗,而是跟房產(chǎn)中介溝通,中介再來(lái)跟賣家溝通和蚪,最后的結(jié)果反映給買家止状。
1. 靜態(tài)代理
靜態(tài)代理就是需要程序員自己編寫一個(gè)代理烹棉,在編譯時(shí)這個(gè)類就已經(jīng)存在了。
示例代碼如下:
- 首先定義功能
interface Something{
public void doSomething();
}
- 定義一個(gè)實(shí)現(xiàn)類
class Shopping implements Something{
@Override
public void doSomething() {
System.out.println("I will go shop!");
}
}
- 為實(shí)現(xiàn)類定義一個(gè)代理類
public class StaticAgent implements Something{
private Shopping shopping;
public StaticAgent(Shopping sp){
shopping = sp;
}
@Override
public void doSomething(){
System.out.println("before I go shopping");
shopping.doSomething();
System.out.println("shoping is done");
}
}
- 具體調(diào)用
public static void main(String[] args) {
Something agent = new StaticAgent(new Shopping());
agent.doSomething();
}
那么就有同學(xué)要問(wèn)了怯疤,這個(gè)代理類好像并沒有存在的必要啊浆洗。但是其實(shí)是有的,一般業(yè)務(wù)只是實(shí)現(xiàn)要實(shí)現(xiàn)的功能集峦,但是如果有些操作需要在業(yè)務(wù)之外做伏社,代理類就可以發(fā)揮作用了。就像我在代理類中寫的那樣塔淤,在執(zhí)行業(yè)務(wù)代碼前后增加了兩條輸出摘昌。同樣可以用于執(zhí)行其他的諸如日志記錄、事務(wù)管理等高蜂。
2. 動(dòng)態(tài)代理
前面所說(shuō)的靜態(tài)代理實(shí)現(xiàn)起來(lái)很簡(jiǎn)單聪黎,但是有個(gè)缺點(diǎn)就是所有的代理類都必須在代碼中寫出來(lái)。那假如很多類都需要增加相同的日志管理备恤,就要針對(duì)所有的類都添加一個(gè)代理類稿饰,這樣就有很多冗余的代碼。動(dòng)態(tài)代理的出現(xiàn)就解決了這個(gè)問(wèn)題露泊,它可以在運(yùn)行時(shí)動(dòng)態(tài)生成代理類喉镰,這樣就可以為一類相同的代理只提供一個(gè)代理類,大大的減少了代碼量滤淳。
動(dòng)態(tài)代理主要有兩種形式梧喷,一種是java本身提供的,使用InvocationHandler接口脖咐。另一種是使用cglib庫(kù)來(lái)實(shí)現(xiàn)铺敌。前者是java利用反射獲取需要實(shí)現(xiàn)的接口然后運(yùn)行時(shí)動(dòng)態(tài)生成代理實(shí)現(xiàn)類,后者則比較強(qiáng)大了屁擅,是利用ASM字節(jié)碼操作框架來(lái)直接修改字節(jié)碼從而生成代理類偿凭。后者速度要比前者快很多,而且可以對(duì)接口進(jìn)行代理派歌。但是無(wú)法為final方法進(jìn)行代理弯囊。
本文先主要描述如何利用java提供的接口實(shí)現(xiàn)動(dòng)態(tài)代理。
java提供的動(dòng)態(tài)代理需要實(shí)現(xiàn)接口InvocationHandler.
public class AgentExecute implements InvocationHandler {
private Object target;
/**
* 綁定對(duì)象
* @param source
* @return
*/
public Object bind(Object source) {
target = source;
return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(),
this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object result = method.invoke(target, args);
return result
}
}
其中Proxy.newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)可以為目標(biāo)類的所有接口創(chuàng)建代理類胶果。
loader為目標(biāo)類的類加載器匾嘱。
interfaces為目標(biāo)類實(shí)現(xiàn)的所有接口。
h為代理類早抠。一般為本實(shí)現(xiàn)類霎烙。
invoke函數(shù)為代理類調(diào)用的方法,當(dāng)代理類調(diào)用需要實(shí)現(xiàn)的函數(shù)時(shí),會(huì)利用反射動(dòng)態(tài)的調(diào)用函數(shù)悬垃∮沃纾可以在調(diào)用該函數(shù)前后增加代碼。
具體調(diào)用代碼為:
AgentExecute agent = new AgentExecute();
Work work = (Work)agent.bind(new DayWork());
work.doWork("tiang", 18);