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

