Spring?AOP手寫動(dòng)態(tài)代理代碼實(shí)例
Spring AOP動(dòng)態(tài)代理
基礎(chǔ)知識(shí)
AOP我們知道,是在不修改源代碼的情況下,為代碼添加一些新功能的技術(shù)。通過動(dòng)態(tài)代理,可以在不修改原始類代碼的前提下,對(duì)方法進(jìn)行攔截和增強(qiáng)。 動(dòng)態(tài)代理常用于在不改變?cè)袠I(yè)務(wù)邏輯的情況下,對(duì)方法的調(diào)用進(jìn)行額外的處理,比如日志記錄、性能監(jiān)控、事務(wù)管理等。它實(shí)現(xiàn)了面向切面編程(AOP)的核心概念。
1、動(dòng)態(tài)代理
AOP其實(shí)就是先通過“一些辦法”拿到“代理對(duì)象”,再對(duì)這個(gè)代理對(duì)象添加新方法。
動(dòng)態(tài)代理的兩種辦法:
1.1 JDK代理:基于接口的動(dòng)態(tài)代理技術(shù)
方法:通過目標(biāo)接口,創(chuàng)建接口實(shí)現(xiàn)類,動(dòng)態(tài)地在運(yùn)行時(shí)內(nèi)存里生成代理對(duì)象,其擁有目標(biāo)對(duì)象的原方法實(shí)現(xiàn),最后通過代理對(duì)象來增加新功能。
看文字有些抽象,現(xiàn)在手動(dòng)用代碼實(shí)現(xiàn)一個(gè)JDK代理的AOP技術(shù),根據(jù)jdk1.8手冊(cè)找到Proxy類,通過newInstance方法實(shí)現(xiàn)動(dòng)態(tài)代理對(duì)象的創(chuàng)建。
“Proxy提供了創(chuàng)建動(dòng)態(tài)代理類和實(shí)例的靜態(tài)方法,它也是由這些方法創(chuàng)建的所有動(dòng)態(tài)代理類的超類。”
(1)準(zhǔn)備接口和實(shí)現(xiàn)類
//=========================接口============================ public interface UserDao { public int add(int a,int b); public String update(String id); } //=========================實(shí)現(xiàn)類============================ public class UserDaoImpl implements UserDao{ @Override public int add(int a, int b) { System.out.println("add......."); return a+b; } @Override public String update(String id) { System.out.println("update==========="); return id; } }
(2)創(chuàng)建UserDao的代理對(duì)象,調(diào)用增強(qiáng)方法 需要注意:InvocationHandler接口里有invoke方法必須實(shí)現(xiàn),而調(diào)用代理對(duì)象的任何方法,實(shí)質(zhì)都是在執(zhí)行invoke方法。
public class JDKProxy { public static void main(String[] args) { /** newProxyInstance的三個(gè)內(nèi)容:類加載器、要實(shí)現(xiàn)的接口的class、 和InvocationHandler(這是個(gè)接口,里面寫的增強(qiáng)類的具體邏輯) */ Class[] interfaces = {UserDao.class}; UserDaoImpl userDao = new UserDaoImpl(); UserDao dao = (UserDao) Proxy.newProxyInstance( JDKProxy.class.getClassLoader(), interfaces, new UserDaoProxy(userDao));//生成的“dao”就是代理對(duì)象實(shí)例。 //也可以直接用匿名內(nèi)部類的方法來寫(new InvocationHandler) int add = dao.add(1, 2);//使用代理對(duì)象dao調(diào)用add的時(shí)候,無需改變add源碼,卻增強(qiáng)了功能。 System.out.println("add=" + add); } } //創(chuàng)建代理對(duì)象代碼 class UserDaoProxy implements InvocationHandler { //把創(chuàng)建的是誰的代理對(duì)象,把誰傳遞過來 //有參數(shù)構(gòu)造傳遞 private Object obj;//要?jiǎng)?chuàng)建obj的代理對(duì)象 public UserDaoProxy(Object obj) { this.obj = obj; } //增強(qiáng)的邏輯:InvocationHandler接口里有invoke方法必須實(shí)現(xiàn) //調(diào)用代理對(duì)象的任何方法,實(shí)質(zhì)都是在執(zhí)行invoke方法。 @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //方法之前(增強(qiáng)) System.out.println("方法之前執(zhí)行...."+method.getName()+" :傳遞的參數(shù)..." + Arrays.toString(args)); //被增強(qiáng)的方法執(zhí)行(原模塊功能不變) Object res = method.invoke(obj, args);//res=3 //方法之后(增強(qiáng)) System.out.println("方法之后執(zhí)行...."+obj); return res; } }
但是有個(gè)遺留問題,invoke函數(shù)究竟是如何被調(diào)用的,往下看了源碼也沒有分析出來,個(gè)人推測(cè)是這樣的:
因?yàn)閭魅氲膐bj其實(shí)就是要?jiǎng)討B(tài)代理的原對(duì)象,但我們不知道具體執(zhí)行什么方法,此時(shí)dao還未調(diào)用add方法。但調(diào)用了以后,invoke方法卻能夠自動(dòng)調(diào)用、增強(qiáng)方法,內(nèi)部應(yīng)該是采用反射的方法,拿到“add”這個(gè)方法名稱,通過getMethod反射出方法的類對(duì)象,再用method.invoke(具體對(duì)象)來實(shí)現(xiàn)方法的調(diào)用,從而調(diào)用add,并完成前后的增強(qiáng)。
1.2 cglib:基于父類的動(dòng)態(tài)代理技術(shù)
方法:為目標(biāo)對(duì)象動(dòng)態(tài)地生成子對(duì)象,其具有了父類的方法(實(shí)際不是繼承) 實(shí)現(xiàn)類似,此處不做代碼示例。
到此這篇關(guān)于Spring AOP手寫動(dòng)態(tài)代理代碼實(shí)例的文章就介紹到這了,更多相關(guān)Spring AOP動(dòng)態(tài)代理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringCloud負(fù)載均衡實(shí)現(xiàn)定向路由詳情
這篇文章主要介紹了SpringCloud負(fù)載均衡實(shí)現(xiàn)定向路由詳情,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-08-08Springboot啟動(dòng)后立即某個(gè)執(zhí)行方法的四種方式
spring項(xiàng)目如何在啟動(dòng)項(xiàng)目是執(zhí)行一些操作,在spring中能通過那些操作實(shí)現(xiàn)這個(gè)功能呢,下面這篇文章主要給大家介紹了關(guān)于Springboot啟動(dòng)后立即某個(gè)執(zhí)行方法的四種方式,需要的朋友可以參考下2022-06-06Java Swing組件布局管理器之FlowLayout(流式布局)入門教程
這篇文章主要介紹了Java Swing組件布局管理器之FlowLayout(流式布局),結(jié)合實(shí)例形式分析了Swing組件布局管理器FlowLayout流式布局的常用方法及相關(guān)使用技巧,需要的朋友可以參考下2017-11-11RabbitMQ消息隊(duì)列的目錄結(jié)構(gòu)
這篇文章主要介紹了RabbitMQ消息隊(duì)列的目錄結(jié)構(gòu),RabbitMQ?屬于消息中間件,主要用于組件之間的解耦,消息的發(fā)送者無需知道消息使用者的存在,反之亦然,那么用了那么久RabbitMQ,其目錄結(jié)構(gòu)是怎樣的呢,讓我們一起來看一下吧2023-08-08基于自定義BufferedReader中的read和readLine方法
下面小編就為大家分享一篇基于自定義BufferedReader中的read和readLine方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2017-12-12