Java動態(tài)代理語法Proxy類原理詳解
1、前言
寫動態(tài)代理的代碼涉及了一個非常重要的類 Proxy,通過Proxy的靜態(tài)方法newProxyInstance才會動態(tài)創(chuàng)建代理對象。
2、newProxyInstance方法
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
三個參數(shù)分別表示: loader表示類加載器, interfaces表示代碼要用來代理的接口 , h表示一個 InvocationHandler 對象,前面兩個參數(shù)容易理解,
最后一個InvocationHandler是什么?
InvocationHandler是一個接口,官方文檔解釋說,每個代理的實例都有一個與之關(guān)聯(lián)的 InvocationHandler 實現(xiàn)類,如果代理的方法被調(diào)用,那么代理便會通知和轉(zhuǎn)發(fā)給內(nèi)部的 InvocationHandler 實現(xiàn)類,由它決定處理。
public interface InvocationHandler { public Object invoke(Object proxy, Method method, Object[] args) throws Throwable; }
InvocationHandler 內(nèi)部只有一個 invoke() 方法,正是這個方法決定了怎么樣處理代理傳遞過來的方法調(diào)用。其中參數(shù)proxy表示代理對象,method表示代理對象調(diào)用的方法,args表示調(diào)用的方法中的參數(shù)。所以Proxy動態(tài)產(chǎn)生的代理會調(diào)用InvocationHandler實現(xiàn)類,所以InvocationHandler才是實際執(zhí)行者。
3、代碼實例
//抽象主題 interface AbstractSubject { void request(); } //真實主題 class RealSubject implements AbstractSubject { public void request() { System.out.println("訪問真實主題方法..."); } } //真實主題 class RealSubject1 implements AbstractSubject { public void request() { System.out.println("訪問真實主題方法1..."); } } //動態(tài)代理類 class DynamicProxy implements InvocationHandler { private Object object; public DynamicProxy(Object object) { this.object = object; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { before(); Object result = method.invoke(object, args); after(); return result; } private void before() { System.out.println("hello!"); } private void after() { System.out.println("bye!"); } } //測試類 public class TestDynamicProxyPattern { public static void main(String[] args) { AbstractSubject abstractSubject = new RealSubject(); DynamicProxy dynamicProxy = new DynamicProxy(abstractSubject); Proxy.newProxyInstance(abstractSubject.getClass().getClassLoader(), abstractSubject.getClass().getInterfaces() , dynamicProxy); abstractProxy.request(); } }
上述動態(tài)代理只需要傳入需要被代理類的對象(DynamicProxy dynamicProxy = new DynamicProxy(abstractSubject)),然后調(diào)用Proxy類的工廠方法newProxyInstance去動態(tài)地創(chuàng)建一個代理類,最后調(diào)用代理類的方法便實現(xiàn)了“增強功能”。使用了動態(tài)代理之后,無論有多少類多少方法需要增加邏輯,只需要在使用的時候?qū)㈩悓ο髠魅氲玫酱韺ο?,然后使用代理對象調(diào)用需要增強的方法即可。
所以這時候如果增加一個實現(xiàn)抽象主題的真是主題類,比如說叫做RealSubject1,這個時候只要只要把該類的對象傳入動態(tài)代理類DynamicProxy中,通過接口又可以實現(xiàn)接口AbstractSubject的實現(xiàn)類。
這樣來實現(xiàn):
AbstractSubject abstractSubject1 = new RealSubject1();
DynamicProxy dynamicProxy = new DynamicProxy(abstractSubject1);
4、總結(jié)
1、區(qū)別于靜態(tài)代理的生成代理類,動態(tài)代理的代理類通過 Proxy.newInstance() 方法生成。靜態(tài)代理和動態(tài)代理的區(qū)別是在于要不要開發(fā)者自己定義 Proxy 類。
2、不管是靜態(tài)代理還是動態(tài)代理,代理與被代理者都要實現(xiàn)接口,還是要是面向接口編程,目的都是增強現(xiàn)有功能。
3、動態(tài)代理通過 Proxy 動態(tài)生成 proxy class,但是它也指定了一個 InvocationHandler 的實現(xiàn)類。
4、動態(tài)代理也有缺陷,它要求需要代理的對象必須實現(xiàn)了某個接口,而且也不夠靈活,動態(tài)代理會為接口中的聲明的所有方法添加上相同的代理邏輯。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java HtmlEmail 郵件發(fā)送的簡單實現(xiàn)代碼
下面小編就為大家?guī)硪黄狫ava HtmlEmail 郵件發(fā)送的簡單實現(xiàn)代碼。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-06-06spring boot使用logback實現(xiàn)多環(huán)境日志配置詳解
這篇文章主要介紹了spring boot使用logback實現(xiàn)多環(huán)境日志配置詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-08-08Java實現(xiàn)終止線程池中正在運行的定時任務(wù)
本篇文章給大家分享了JAVA中實現(xiàn)終止線程池中正在運行的定時任務(wù)的具體步驟和方法,有需要的朋友跟著學(xué)習(xí)下。2018-05-05SpringBoot整合screw實現(xiàn)自動生成數(shù)據(jù)庫設(shè)計文檔
使用navicat工作的話,導(dǎo)出的格式是excel不符合格式,還得自己整理。所以本文將用screw工具包,整合到springboot的項目中便可以自動生成數(shù)據(jù)庫設(shè)計文檔,非常方便,下面就分享一下教程2022-11-11在springboot中實現(xiàn)個別bean懶加載的操作
這篇文章主要介紹了在springboot中實現(xiàn)個別bean懶加載的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-10-10