看了好多關於代理的文章,理解和整理一下。 1、代理的基本構成 抽象角色:聲明真實對象和代理對象的共同介面,這樣可在任何使用真實對象的地方都可以使用代理對象。 代理角色:代理對象內部含有真實對象的引用,從而可以在任何時候操作真實對象。代理對象提供一個與真實對象相同的介面,以便可以在任何時候替代真實對象 ...
看了好多關於代理的文章,理解和整理一下。
1、代理的基本構成
抽象角色:聲明真實對象和代理對象的共同介面,這樣可在任何使用真實對象的地方都可以使用代理對象。
代理角色:代理對象內部含有真實對象的引用,從而可以在任何時候操作真實對象。代理對象提供一個與真實對象相同的介面,以便可以在任何時候替代真實對象。代理對象通常在客戶端調用傳遞給真實對象之前或之後,執行某個操作,而不是單純地將調用傳遞給真實對象,同時,代理對象可以在執行真實對象操作時,附加其他的操作,相當於對真實對象進行封裝。
真實角色:即為代理對象所代表的目標對象,代理角色所代表的真實對象,是我們最終要引用的對象。
下圖有三種角色:Subject抽象角色、RealSubject真實角色、Proxy代理角色。其中:Subject角色負責定義RealSubject和Proxy角色應該實現的介面;RealSubject角色用來真正完成業務服務功能;Proxy角色負責將自身的request請求,調用RealSubject對應的request功能來實現業務功能,自己不真正做業務。
2、靜態代理
interface Subject//抽象角色 { public void doSomething(); } class RealSubject implements Subject//真實角色 { public void doSomething() { System.out.println( "call doSomething()" ); } } class SubjectProxy implements Subject//代理角色 { //代理模式的作用是:為其他對象提供一種代理以控制對這個對象的訪問。 Subject subimpl = new RealSubject(); public void doSomething() { System.out.println("before"); //調用目標對象之前可以做相關操作 subimpl.doSomething(); System.out.println("after");//調用目標對象之後可以做相關操作 } } public class Test { public static void main(String[] args) throws Exception { Subject sub = new SubjectProxy(); sub.doSomething(); } }
可以看到,SubjectProxy實現了Subject介面(和RealSubject實現相同介面),並持有的是Subject介面類型的引用。這樣調用的依然是doSomething方法,只是實例化對象的過程改變了,結果來看,代理類SubjectProxy可以自動為我們加上了before和after等我們需要的動作。
如果將來需要實現一個新的介面,就需要在代理類里再寫該介面的實現方法,對導致代理類的代碼變得臃腫;另一方面,當需要改變抽象角色介面時,無疑真實角色和代理角色也需要改變。
3、JDK動態代理
interface Subject { public void doSomething(); } class RealSubject implements Subject { public void doSomething() { System.out.println( "call doSomething()" ); } } class ProxyHandler implements InvocationHandler { private Object tar; //綁定委托對象,並返回代理類 public Object bind(Object tar) { this.tar = tar; //綁定該類實現的所有介面,取得代理類 return Proxy.newProxyInstance(tar.getClass().getClassLoader(), tar.getClass().getInterfaces(), this); } public Object invoke(Object proxy , Method method , Object[] args)throws Throwable//不依賴具體介面實現 { Object result = null;//被代理的類型為Object基類 //這裡就可以進行所謂的AOP編程了 //在調用具體函數方法前,執行功能處理 result = method.invoke(tar,args); //在調用具體函數方法後,執行功能處理 return result; } } public class Test { public static void main(String args[]) { ProxyHandler proxy = new ProxyHandler(); //綁定該類實現的所有介面 Subject sub = (Subject) proxy.bind(new RealSubject()); sub.doSomething(); } }
在調用過程中使用了通用的代理類包裝了RealSubject實例,然後調用了Jdk的代理工廠方法實例化了一個具體的代理類。最後調用代理的doSomething方法,還有附加的before、after方法可以被任意復用(只要我們在調用代碼處使用這個通用代理類去包裝任意想要需要包裝的被代理類即可)。當介面改變的時候,雖然被代理類需要改變,但是我們的代理類卻不用改變了。這個調用雖然足夠靈活,可以動態生成一個具體的代理類,而不用自己顯示的創建一個實現具體介面的代理類。