詳細(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)品,并拿到錢(qián):"+money);
}
public void AfterService(float money) {
System.out.println("提供售后服務(wù),并拿到錢(qián):"+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)品,并拿到錢(qián):"+money);
}
/**
* 售后
* @param money
*/
public void afterService(float money){
System.out.println("提供售后服務(wù),并拿到錢(qián):"+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-02
Java后臺(tái)基于POST獲取JSON格式數(shù)據(jù)
這篇文章主要介紹了Java后臺(tái)基于POST獲取JSON格式數(shù)據(jù),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03
Spring 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-07
springcloud整合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-06
java使用renderer將pdf按頁(yè)轉(zhuǎn)換為圖片
這篇文章主要為大家詳細(xì)介紹了java使用renderer將pdf按頁(yè)轉(zhuǎn)換為圖片,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-12-12

