詳細(xì)分析java 動態(tài)代理
1、動態(tài)代理的特點(diǎn):
字節(jié)碼隨用隨創(chuàng)建,隨用隨加載
2、作用:
不修改源碼的基礎(chǔ)上對源碼進(jìn)行加強(qiáng)
3、分類:
(1)基于接口的動態(tài)代理:
涉及到的類:Proxy,由JDK官方提供,使用Proxy類中的newProxyInstance方法創(chuàng)建對象。創(chuàng)建代理對象時(shí)要求被代理對象至少實(shí)現(xiàn)一個(gè)接口,否則無法使用
參數(shù):
- ClassLoader:類加載器,他是用于加載對象字節(jié)碼的,和被代理對象使用相同的類加載器,為固定寫法
- class[]:字節(jié)碼數(shù)組,他是用于讓代理對象和被代理對象具有相同的方法,也是固定寫法
- InvocationHandler:用戶提供增強(qiáng)的代碼 ,他是讓我們寫如何代理。我們一般都是寫一個(gè)該接口的實(shí)現(xiàn)類,通常情況下都是匿名內(nèi)部類,但不是必須的,此接口的實(shí)現(xiàn)類都是誰用誰寫
示例: 創(chuàng)建 Producter接口和實(shí)體類
package com.mingqi.proxy; /** * 對生產(chǎn)廠家要求的接口 */ public interface IProducer { /** * 銷售 * @param money */ public void SaleProduct(float money); /** * 售后 * @param money */ public void AfterService(float money); }
package com.mingqi.proxy; public class Producer implements IProducer { public void SaleProduct(float money) { System.out.println("銷售產(chǎn)品,并拿到錢:"+money); } public void AfterService(float money) { System.out.println("提供售后服務(wù),并拿到錢:"+money); } }
測試方法:
public static void main(String[] args) { /* 1、動態(tài)代理 特點(diǎn):字節(jié)碼隨用隨創(chuàng)建,隨用隨加載 作用:不修改源碼的基礎(chǔ)上對源碼進(jìn)行加強(qiáng) 分類:基于接口的動態(tài)代理 涉及的類: Proxy 提供者:JDK官方 如何創(chuàng)建代理對象: 使用Proxy類中的newProxyInstance方法 創(chuàng)建代理對象的要求: 被代理對象至少實(shí)現(xiàn)一個(gè)接口,如果沒有則不能使用 newProxyInstance的方法參數(shù): ClassLoader:類加載器 他是用于加載代理對象字節(jié)碼的,和被代理對象使用相同的類加載器,固定寫法 class[] :字節(jié)碼數(shù)組 InvocationHandler 用于提供增強(qiáng)的代碼 他是讓我們寫如何代理,我們一般都是寫一個(gè)接口的實(shí)現(xiàn)類,通常情況下都是匿名內(nèi)部類,但不是必須的,此接口的實(shí)現(xiàn)類都是誰用誰寫*/ final Producer producer=new Producer(); IProdurcer proxyProducer= (IProdurcer)Proxy.newProxyInstance(producer.getClass().getClassLoader(), producer.getClass().getInterfaces(), new InvocationHandler() { /** * 作用:執(zhí)行被代理對象的任何接口方法都會經(jīng)過該方法 * 方法參數(shù)的含義 * @param proxy 代理對象的引用 * @param method 當(dāng)前執(zhí)行的方法 * @param args 當(dāng)前執(zhí)行方法所需的參數(shù) * @return 被代理對象有相同的返回值 * @throws Throwable */ public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //提供增強(qiáng)的代碼: Object returnValue=null; //1、獲取方法執(zhí)行的參數(shù) Float money=(Float) args[0]; //判斷當(dāng)前方法是否是銷售 if("SaleProduct".equals(method.getName())) { returnValue=method.invoke(producer,money*0.8f); } return returnValue; } }); proxyProducer.SaleProduct(10000f); }
(2) 基于子類的動態(tài)代理
涉及到的類:Enhancer,由第三方cglib提供,使用Enhancer類中的create方法創(chuàng)建對象。創(chuàng)建代理對象的類不能是最終類,否則無法使用
參數(shù):
- Class:字節(jié)碼,他是用于指定被代理對象的字節(jié)碼,為固定寫法
- Callback:用戶提供增強(qiáng)的代碼 ,他是讓我們寫如何代理。我們一般都是寫一個(gè)該接口的實(shí)現(xiàn)類,通常情況下都是匿名內(nèi)部類,但不是必須的,此接口的實(shí)現(xiàn)類都是誰用誰寫,我們一般寫的都是該接口的子接口實(shí)現(xiàn)類MethodInterceptor
示例: 創(chuàng)建 Product接口和實(shí)體類
package com.mingqi.cglib; /** * 一個(gè)生產(chǎn)者 */ public class Product { /** * 銷售 * @param money */ public void saleProduct(float money){ System.out.println("銷售產(chǎn)品,并拿到錢:"+money); } /** * 售后 * @param money */ public void afterService(float money){ System.out.println("提供售后服務(wù),并拿到錢:"+money); } }
測試類及方法:
package com.mingqi.cglib; import net.sf.cglib.proxy.Enhancer; import net.sf.cglib.proxy.MethodInterceptor; import net.sf.cglib.proxy.MethodProxy; import java.lang.reflect.Method; public class Client { public static void main(String[] args) { final Product product=new Product(); Product cglibproduct= (Product) Enhancer.create(product.getClass(), new MethodInterceptor() { public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { Object returnValue=null; //1、獲取方法執(zhí)行的參數(shù) Float money=(Float)objects[0]; //判斷當(dāng)前方法是不是銷售 if("SaleProduct".equals(method.getName())) { returnValue = method.invoke(product, money*0.8f); } return returnValue; } }); cglibproduct.SaleProduct(1000f); } }
以上就是創(chuàng)建動態(tài)代理對象的兩種類型,以后要經(jīng)常練習(xí)使用,讓這種思想能給我們工作中帶來方便。
到此這篇關(guān)于詳細(xì)分析java 動態(tài)代理的文章就介紹到這了,更多相關(guān)java 動態(tài)代理內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Java JDK動態(tài)代理(AOP)用法及實(shí)現(xiàn)原理詳解
- Java兩種方式實(shí)現(xiàn)動態(tài)代理
- 詳解Java JDK動態(tài)代理
- 詳解Java Cglib動態(tài)代理
- Java簡單實(shí)現(xiàn)動態(tài)代理模式過程解析
- Java JDK動態(tài)代理實(shí)現(xiàn)原理實(shí)例解析
- Java動態(tài)代理語法Proxy類原理詳解
- Java動態(tài)代理靜態(tài)代理實(shí)例分析
- Java代理模式實(shí)例詳解【靜態(tài)代理與動態(tài)代理】
- JAVA使用動態(tài)代理對象進(jìn)行敏感字過濾代碼實(shí)例
- Java動態(tài)代理模式的深入揭秘
- Java 動態(tài)代理的多種實(shí)現(xiàn)方式
相關(guān)文章
Spring自動裝配Bean實(shí)現(xiàn)過程詳解
這篇文章主要介紹了Spring自動裝配Bean實(shí)現(xiàn)過程詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-02-02Java后臺基于POST獲取JSON格式數(shù)據(jù)
這篇文章主要介紹了Java后臺基于POST獲取JSON格式數(shù)據(jù),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03Spring Boot詳解創(chuàng)建和運(yùn)行基礎(chǔ)流程
這篇文章主要介紹了SpringBoot創(chuàng)建和運(yùn)行的基礎(chǔ)流程,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-06-06springcloud整合seata的實(shí)現(xiàn)代碼
這篇文章主要介紹了springcloud整合seata的實(shí)現(xiàn)方法,整合步驟通過引入spring-cloud-starter-alibaba-seata?jar包,文中結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-05-05詳解Java并發(fā)包中線程池ThreadPoolExecutor
ThreadPoolExecutor是Java語言對于線程池的實(shí)現(xiàn)。線程池技術(shù)使線程在使用完畢后不回收而是重復(fù)利用。如果線程能夠復(fù)用,那么我們就可以使用固定數(shù)量的線程來解決并發(fā)問題,這樣一來不僅節(jié)約了系統(tǒng)資源,而且也會減少線程上下文切換的開銷2021-06-06java使用renderer將pdf按頁轉(zhuǎn)換為圖片
這篇文章主要為大家詳細(xì)介紹了java使用renderer將pdf按頁轉(zhuǎn)換為圖片,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-12-12