基於介面的動態代理
阿新 • • 發佈:2020-12-19
故事背景:
以前,生產商生產電腦賣給消費者,生產商直接跟消費者對接。
現在,生產商先把電腦賣給經銷商,經銷商再把電腦賣給消費者,消費者付給經銷商1000元,經銷商要從中拿走20%,剩餘的錢再給生產商
第一步:給生產者制定一個規範,要求生產者必須有哪些功能
/**
* 對生產廠家要求的介面
*/
public interface IProducer {
/**
* 銷售
* @param money
*/
public void saleProduct(Float money);
/**
* 售後
* @param money
*/
public void afterService(Float money);
}
第二步:新建一個生產商的實體類
/**
* 一個生產者
*/
public class Producer implements IProducer {
/**
* 銷售
* @param money
*/
public void saleProduct(Float money) {
System.out.println("銷售產品,並拿到錢:"+money);
}
/**
* 售後
* @param money
*/
public void afterService(Float money) {
System.out.println("提供售後服務,並拿到錢:"+money);
}
}
第三步:實現基於介面的動態代理
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
* 模擬一個消費者
*/
public class Client {
public static void main(String[] args) {
final Producer producer= new Producer();
/**
* 動態代理
* 特點:位元組碼隨用隨建立,隨用隨載入
* 作用:不修改原始碼的基礎上對方法增強
* 分類:基於介面的動態代理
* 基於子類的動態代理
* 基於介面的動態代理:
* 涉及的類:Proxy
* 提供者:JDK官方
* 如何建立代理物件
* 使用proxy類中的newProxyInstance方法
* 建立代理物件的要求:
* 被代理類最少實現一個介面,如果沒有則不能使用
* newProxyInstance方法的引數
* ClassLoader 類載入器
* 他是用於載入代理物件位元組碼的,和被代理物件使用相同的類載入器,代理誰就寫誰的類載入器,固定寫法
* Class[] 位元組碼陣列
* 他是用於讓代理物件和被代理物件有相同方法,固定寫法
* InvocationHandler 用於提供增強的程式碼
* 他是讓我們寫如何代理,我們一般都是寫一個介面的實現類,通常情況下都是匿名內部類,但不是必須的
* 此介面的實現類都是誰用誰寫
*/
IProducer proxyProducer = (IProducer) Proxy.newProxyInstance(producer.getClass().getClassLoader(),
producer.getClass().getInterfaces(), new InvocationHandler() {
/**
* 作用:執行被代理物件的任何介面方法都會經過該方法
* 方法引數的含義
* @param proxy 代理物件的引用
* @param method 當前執行的方法
* @param args 當前執行方法所需的引數
* @return 和被代理物件方法有相同的返回值
* @throws Throwable
*/
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//提供增強程式碼
Object returnValue = null;
// 1、獲取方法執行的引數
Float money = (Float)args[0];
//2、判斷當前方法是不是銷售
if ("saleProduct".equals(method.getName())) {
returnValue = method.invoke(producer,money*0.8f);
}
return returnValue;
}
});
proxyProducer.saleProduct(1000f);
}
}
執行結果: