詳細(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ì)象使用相同的類加載器,為固定寫法
- class[]:字節(jié)碼數(shù)組,他是用于讓代理對(duì)象和被代理對(duì)象具有相同的方法,也是固定寫法
- InvocationHandler:用戶提供增強(qiáng)的代碼 ,他是讓我們寫如何代理。我們一般都是寫一個(gè)該接口的實(shí)現(xiàn)類,通常情況下都是匿名內(nèi)部類,但不是必須的,此接口的實(shí)現(xiàn)類都是誰(shuí)用誰(shuí)寫
示例: 創(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ì)象使用相同的類加載器,固定寫法
class[] :字節(jié)碼數(shù)組
InvocationHandler 用于提供增強(qiáng)的代碼
他是讓我們寫如何代理,我們一般都是寫一個(gè)接口的實(shí)現(xiàn)類,通常情況下都是匿名內(nèi)部類,但不是必須的,此接口的實(shí)現(xiàn)類都是誰(shuí)用誰(shuí)寫*/
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é)碼,為固定寫法
- Callback:用戶提供增強(qiáng)的代碼 ,他是讓我們寫如何代理。我們一般都是寫一個(gè)該接口的實(shí)現(xiàn)類,通常情況下都是匿名內(nèi)部類,但不是必須的,此接口的實(shí)現(xiàn)類都是誰(shuí)用誰(shuí)寫,我們一般寫的都是該接口的子接口實(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)文章
springboot組件初始化后的4種啟動(dòng)方式及常用方法
在Spring Boot中,您可以通過(guò)幾種方式在組件初始化后執(zhí)行啟動(dòng)任務(wù),下面小編給大家分享springboot組件初始化后的4種啟動(dòng)方式及常用方法,感興趣的朋友一起看看吧2024-06-06
mybatis plus 的動(dòng)態(tài)表名的配置詳解
這篇文章主要介紹了mybatis plus 的動(dòng)態(tài)表名的配置詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09
SpringCloud分布式事務(wù)Seata部署和集成過(guò)程
這篇文章主要介紹了SpringCloud分布式事務(wù)Seata部署和集成過(guò)程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-10-10
淺談SpringCloud feign的http請(qǐng)求組件優(yōu)化方案
這篇文章主要介紹了淺談SpringCloud feign的http請(qǐng)求組件優(yōu)化方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-02-02
實(shí)現(xiàn)一個(gè)基于Servlet的hello world程序詳解步驟
Java Servlet 是運(yùn)行在 Web 服務(wù)器或應(yīng)用服務(wù)器上的程序,它是作為來(lái)自 Web 瀏覽器或其他 HTTP 客戶端的請(qǐng)求和 HTTP 服務(wù)器上的數(shù)據(jù)庫(kù)或應(yīng)用程序之間的中間層2022-02-02
Java并發(fā)編程之synchronized底層實(shí)現(xiàn)原理分析
這篇文章主要介紹了Java并發(fā)編程之synchronized底層實(shí)現(xiàn)原理,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-02-02
基于SpringBoot實(shí)現(xiàn)HTTP請(qǐng)求簽名驗(yàn)證機(jī)制
在分布式系統(tǒng)交互中,API接口的安全性至關(guān)重要,本文將深入解析基于Spring Boot實(shí)現(xiàn)的HTTP請(qǐng)求簽名驗(yàn)證機(jī)制,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2025-04-04

