在上一篇文章中,我們講到了簡單的代理模式實(shí)現(xiàn)逆日。但是同時(shí)也產(chǎn)生了一個(gè)問題:假如我們需要代理的真實(shí)類中有很多方法哩簿,那么我們就要仿照真實(shí)類覆蓋實(shí)現(xiàn)
所有的方法航邢。這使得我們的代碼就有很多重復(fù)。但是我們可以使用簡單的JDK動態(tài)代理的方式來實(shí)現(xiàn)溯革。
首先贞绳,假設(shè)我們的IDBQuery
有如下實(shí)現(xiàn):
public interface IDBQuery {
public String query();
public String getName();
public int getAge();
public String getData();
}
然后在DBQuery
中進(jìn)行實(shí)現(xiàn)和打出簡單的說明:
public class DBQuery implements IDBQuery {
public DBQuery(){
//start db connect
try {
Thread.sleep(1000);
System.out.println("start db connect...");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
@Override
public String query() {
System.out.println("begin query...");
return "begin query...";
}
@Override
public String getName() {
System.out.println("getName()...");
return null;
}
@Override
public int getAge() {
System.out.println("getAge()");
return 0;
}
@Override
public String getData() {
System.out.println("getData()");
return null;
}
}
接下來我們創(chuàng)建DBQuery
的InvocationHandler
:
public class InvocationDBQueryHandler implements InvocationHandler {
private DBQuery dbQuery = null;
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
if(dbQuery == null){
dbQuery = new DBQuery(); //如果第一次被調(diào)用才生成真實(shí)對象
}
String methodName = method.getName();
System.out.println("called method name:" + methodName);
return method.invoke(dbQuery, args);//轉(zhuǎn)調(diào)真實(shí)方法
}
public static IDBQuery createJdkDBQuery(){
IDBQuery idbq = (IDBQuery) Proxy.newProxyInstance(ClassLoader.getSystemClassLoader(),
new Class[]{IDBQuery.class}, new InvocationDBQueryHandler());
return idbq;
}
}
以下是測試:
IDBQuery jdkIdbq = InvocationDBQueryHandler.createJdkDBQuery();
System.out.println("after create jdkDBQuery");
jdkIdbq.query();
jdkIdbq.getAge();
jdkIdbq.getName();
jdkIdbq.getData();
得到如下輸出結(jié)果:
after create jdkDBQuery
start db connect...
called method name:query
begin query...
called method name:getAge
getAge()
called method name:getName
getName()...
called method name:getData
getData()
從結(jié)果可以看出:
就像上一篇的DBQueryProxy
一樣,我們僅在query
方法或其他方法被調(diào)用時(shí)才初始化DBQuery
致稀。
這種動態(tài)代理的方法簡單快速的實(shí)現(xiàn)了我們需要延遲加載DBQuery
的需求冈闭。其實(shí)動態(tài)代理的常用場景是日志記錄和
業(yè)務(wù)分離,例如Spring Framework中的AOP抖单。使用動態(tài)代理我們可以在被調(diào)用方法的前后執(zhí)行我們需要的操作萎攒。
于2015-06-07