1. 程式人生 > 其它 >基於介面的動態代理

基於介面的動態代理

技術標籤:javajavaproxyaop反射jdk

故事背景:
以前,生產商生產電腦賣給消費者,生產商直接跟消費者對接。
現在,生產商先把電腦賣給經銷商,經銷商再把電腦賣給消費者,消費者付給經銷商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); } }

執行結果:

在這裡插入圖片描述