詳解Java反射實現(xiàn)Aop代理
利用反射生成JDK的動態(tài)代理,也就是AOP中的AOP代理,代替目標(biāo)對象,從而在代碼中織入增強。
定義代理接口
由于JDKf動態(tài)代理只能為接口創(chuàng)建動態(tài)代理,故先定義接口,假定我們需要對數(shù)據(jù)的Save方法添加事務(wù)處理,我們有一個UserDao接口,里面有一個Save方法,代碼如下:
public interface UserDao { public void save(); }
定義代理實現(xiàn)
下面具體來實現(xiàn)接口定義的Save方法,我們采用下面的代碼來實現(xiàn)。
public class UserDaoImpl implements UserDao { @Override public void save() { System.out.println("I am save user...."); } }
定義增強代碼
我們有如下的操作,在保存用戶之前打開事務(wù),在保存用戶之后提交事務(wù),在增強代碼中定義兩個方法before()和after(),分別用在save()方法的執(zhí)行開始之前和執(zhí)行之后。
public class UserTx { public void before(){ System.out.println("before save....."); } public void after(){ System.out.println("after save......"); } }
定義Invocation handler
之所以要定義handler是因為執(zhí)行動態(tài)代理時,實際執(zhí)行的是handler里面的invoke()方法,這樣的話,我們在invoke()方法里面自定義方法的內(nèi)容,從而就達到了代理和增強的邏輯和效果。
public class UserDaoInvocationHandler implements InvocationHandler { / 需要代理的對象 / private Object proxyObj; / 指定我們需要代理的對象 @param proxyObj */ public void setProxyObj(Object proxyObj) { this.proxyObj = proxyObj; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { UserTx tx = new UserTx(); tx.before(); Object resultObj = method.invoke(proxyObj, args); tx.after(); return resultObj; } }
測試結(jié)果
上面已經(jīng)定義好所有的東西,我們就實際來動態(tài)代理我們指定的對象,用代理后的對象來執(zhí)行我們要執(zhí)行的方法,驗證是否代理成功。
import java.lang.reflect.Proxy; public class ProxyTst { public static void main(String[] args) { // proxy object UserDao target = new UserDaoImpl(); // invocation handler UserDaoInvocationHandler handler = new UserDaoInvocationHandler(); handler.setProxyObj(target); // proxy UserDao targeted = (UserDao) Proxy.newProxyInstance( target.getClass().getClassLoader(), target.getClass().getInterfaces(), handler); // execute proxyed object targeted.save(); } }
執(zhí)行上面的代碼之后我們可以在控制臺看見如下的輸出,證明UserDao已經(jīng)被成功代理,同時我們也為我們的程序成功的添加了事務(wù)功能。
before save..... I am save user.... after save......
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java Class.forName()用法和newInstance()方法原理解析
這篇文章主要介紹了Java Class.forName()用法和newInstance()方法原理解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-08-08使用google.kaptcha來生成圖片驗證碼的實現(xiàn)方法
這篇文章主要介紹了使用google.kaptcha來生成圖片驗證碼的實現(xiàn)方法,非常不錯具有一定的參考借鑒價值,需要的朋友可以參考下2018-09-09解決JPA?save()方法null值覆蓋掉mysql預(yù)設(shè)的默認(rèn)值問題
這篇文章主要介紹了解決JPA?save()方法null值覆蓋掉mysql預(yù)設(shè)的默認(rèn)值問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-11-11Maven 搭建SpringMVC+Hibernate項目詳解
本文主要介紹Maven 搭建SpringMVC+Hibernate的知識,這里整理了詳細(xì)的資料,并附示例代碼,有興趣的小伙伴可以參考下2016-09-09解決SpringMVC、tomcat、Intellij idea、ajax中文亂碼問題
這篇文章主要介紹了解決SpringMVC、tomcat、Intellij idea、ajax中文亂碼問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-09-09