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