首頁 > 軟體

Java 深入淺出講解代理模式

2022-03-31 19:00:11

1、動態代理模式

動態代理的特點:

  • 當代理物件的時候,不需要實現介面
  • 代理物件的生成,是利用JDK的API,動態的在記憶體中構建代理物件(需要我們指定建立代理物件/目標物件實現的介面的型別)

動態代理的別稱:JDK代理、介面代理

2、JDK動態代理

類圖:

Java動態代理類位於java.lang.reflect包下

一般主要涉及到以下兩個類:

1、Interface InvocationHandler : 該介面中僅定義了一個方法public object invoke(Object obj,Method method, Object[] args) 在實際使用時,第一個引數obj一般是指代理類,method是被代理的方法,args為該方法的引數陣列。這個抽象方法在代理類中動態實現。

2、Proxy:該類即為動態代理類

static Object newProxyInstance(ClassLoader loader, Class[] interfaces,InvocationHandler h):

返回代理類的一個範例,返回後的代理類可以當作被代理類使用(可使用被代理類的在介面中宣告過的方法)

動態代的實現步驟:

  • 建立一個實現介面InvocationHandler的類,它必須實現invoke方法
  • 建立被代理的類以及介面
  • 呼叫Proxy的靜態方法,建立一個代理類:newProxyInstance(ClassLoader loader,Class[]
  • 通過代理呼叫方法

3、JDK動態代理程式碼演示

比如現在有一個汽車駕駛的方法:

public interface Moveable {
    void move();
}

現在有一輛汽車:

//實現Moveable 介面,並隨機暫停一段時間

import java.util.Random;
public class Car implements Moveable{
    @Override
    public void move() {
        try{
            Thread.sleep(new Random().nextInt(1000));
            System.out.println("汽車行駛中");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

時間代理類:

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class TimeHandler implements InvocationHandler{

    public TimeHandler(Object target){
        super();
        this.target = target;
    }
    private Object target;


    /**
     *
     * @param proxy :被代理的物件
     * @param method:被代理物件的方法
     * @param args:方法的引數
     * @return
     * @throws Throwable
     * 返回值:Object 方法的返回值
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        long startTime = System.currentTimeMillis();
        System.out.println("汽車開始行使");

        method.invoke(target);

        long endTime = System.currentTimeMillis();
        System.out.println("汽車行駛結束,行駛的時間為:"+(endTime-startTime)+"毫秒");
        return null;
    }
}

測試類:

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;

public class Test {
    public static void main(String[] args) {

        Car car = new Car();
        InvocationHandler h = new TimeHandler(car);
        Class<?> cls = car.getClass();

        /**
         * newProxyInstanced的引數
         * 分別是:類載入器、實現的介面、實現的處理器
         */
        Moveable m = (Moveable) Proxy.newProxyInstance(cls.getClassLoader(),
                cls.getInterfaces(),h);
        m.move();
    }
}

這樣的輸出結果是:

汽車開始行使
汽車行駛中
汽車行駛結束,行駛的時間為:137毫秒
//後面的時間是隨機產生的,每次都不一樣

注意:

JDK代理只能代理實現了介面的類,沒有實現介面的不能代理

以上就是JDK動態代理,當然還有cglib動態代理:

cglib是針對類來實現代理的,cglib的原理是對指定的目標類生成一個子類,並覆蓋其中方法實現增強,但因為採用的是繼承,所以不能對final修飾的類進行代理,因為小應學長自己對這一塊也沒有完全掌握,這裡就不多講解,大家可以參考其他博主的技術文章。

三月份完美收官!最後小應學長十分感謝大家的支援的陪伴,你們在支援是我最大的動力,繼續寫出大家喜歡的文章!

到此這篇關於Java 深入淺出講解代理模式的文章就介紹到這了,更多相關Java 代理模式內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com