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