設計模式之代理模式學習

1 代理模式概念介紹

代理模式定義:爲其他對象提供一種代理以控制對這個對象的訪問。代理對象起到中介作用,可去掉功能服務或增加額外的服務。

1.1 常見代理模式

(1)遠程代理:爲不同地理的對象提供局域網代表對象。

(2)虛擬代理:根據需要將資源消耗很大的對象進行延遲真正需要的時候進行創建。

(3)保護代理:控制對一個對象的訪問權限控制。

(4)智能引用代理:提供對目標對象額外服務。

2 商用代理模式原理

2.1 靜態代理概念及實現

靜態代理:代理和被代理對象在代理之前是確定的。他們都實現相同的接口 或者繼承相同的抽象類。

2.2 動態代理

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):返回代理類的一個實例,返回後的代理類可以當做被代理類使用(可使用被代理類的在接口中聲明過的方法)

JDK動態代理(Dynamic Proxy)是這樣一種class:

它是在運行時產生的class

該class需要實現一組interface

使用動態代理類時,必須實現InvocationHandler接口。

2.3 動態代理類實現步驟

(1)創建一個接口InvocationHandler的類,它必須實現invoke方法。
(2)創建被代理的類以及接口

(3)調用Proxy的靜態方法,創建一個代理類

(4)通過代理類調用方法

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

例如:

package com.imooc.jdkproxy;

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;
    /**
     * 參數:
     * proxy 被代理的對象
     * method 被代理的方法
     *  args 方法參數
     *  
     *  返回值 Object對象
     */
    @Override
    public Object invoke(Object arg0, 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;
    }

}

 

package com.imooc.proxy;

public interface Moveable {
    void move();
}
 

package com.imooc.proxy;

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.Proxy;

import com.imooc.proxy.Car;
import com.imooc.proxy.Moveable;

public class Test {
    public static void main(String[] args) {
        Car car = new Car();
        InvocationHandler h = new TimeHandler(car);
        Class<?> cls = car.getClass();
        /**
         * arg0 被代理類的類加載器
         * arg1 實現接口
         * arg2 invocationHandler
         */
        Moveable m =  (Moveable)Proxy.newProxyInstance(cls.getClassLoader(), cls.getInterfaces(), h);
        m.move();
    }
    
}

2.4 使用cglib動態產生代理

2.4.1 JDK動態代理與CGLIB動態代理區別

JDK動態代理:

(1)只能代理實現了接口的類

(2)沒有實現接口的類不能實現JDK的動態代理。

CGLIB動態代理

(1)針對類來實現代理的。

(2)對指定目標類產生一個子類,通過方法攔截技術攔截所有父類方法的調用。

CGLIB動態代理實現原理可參考博客:https://blog.csdn.net/yhl_jxy/article/details/80633194

 

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章