代理模式探祕

代理模式是經常使用的設計模式,通常可分爲靜態和動態代理。java

靜態代理

  • 繼承相同的抽象類
  • 實現相同的接口

動態代理

JDK動態代理

image

實現步驟設計模式

  1. 被代理的類classA以及實現的接口interfaceA
  2. 建立一個事務處理器,實現接口InvocationHandler,以及invoke方法
  3. 調用java.lang.reflect.Proxy的靜態方法,建立一個代理對象
Proxy.newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h);
  1. 經過代理對象調用方法
public class AHandler implements InvocationHandler {

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

	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		
		System.out.println("\n=================");
		method.invoke(target);
		System.out.println("=================");
		
		Class<? extends Object> proxyClass = proxy.getClass();
		System.out.println(" --- proxy class:" + proxyClass.getName());
		Class<?> fooProxyClass = proxyClass.getSuperclass();
		System.out.println(" --- super proxy class:" + fooProxyClass.getName());
		
		System.out.println(" --- method:" + method.getName());
		
//		如下是一個死循環
//		Method[] methods = proxyClass.getDeclaredMethods();
//		for (Method m : methods){
//			System.out.println(m.getName());
//			if("say".equals(m.getName())){
//				System.out.println(" --- 讓$Proxy0這個代理對象調用say方法");
//				m.invoke(proxy);
//			}			
//		}
		
		return 2;
	}

}
public class A implements InterA{

	@Override
	public void say() {
		System.out.println(" --- i love m --- ");
	}

	@Override
	public int hear() {
		System.out.println(" --- Saturday --- ");
		return 1;
	}

	public static void main(String[] args) {
		
		A a = new A();
		Class<? extends A> aclass = a.getClass();
		
		AHandler handler = new AHandler(a);
		
		Object proxy = Proxy.newProxyInstance(aclass.getClassLoader(), aclass.getInterfaces(), handler);
		System.out.println(" --- return proxy class:" + proxy.getClass().getName());
		
		InterA aProxy = (InterA) proxy;
		aProxy.say();
		
		int i = aProxy.hear();
		System.out.println(i);
		
		int h = aProxy.hashCode();
		System.out.println(h);
	}
	
}
//console
 --- return proxy class:com.sun.proxy.$Proxy0

=================
 --- i love m --- 
=================
 --- proxy class:com.sun.proxy.$Proxy0
 --- super proxy class:java.lang.reflect.Proxy
 --- method:say

=================
 --- Saturday --- 
=================
 --- proxy class:com.sun.proxy.$Proxy0
 --- super proxy class:java.lang.reflect.Proxy
 --- method:hear
2

=================
=================
 --- proxy class:com.sun.proxy.$Proxy0
 --- super proxy class:java.lang.reflect.Proxy
 --- method:hashCode
2

以上是根據視頻本身寫的測試類,關於JDK動態代理,總結以下:ide

  • 代理對象會代理接口中定義的全部方法,以及從Object繼承過來的方法
  • 根據JDK規定,代理以$Proxy開頭
  • 代理類父類是java.lang.reflect.Proxy
  • Handler中必須是method.invoke(target),不能是method.invoke(proxy),雖然如今不清楚proxy參數時什麼用,但它所代理的方法就是去handler中調用invole方法,即say.invoke(proxy),必然死循環
  • 代理方法返回值即invoke的返回值

小結

JDK動態代理的實現原理,基本上就是反射,以say()爲例,代理對象調用say(),實際上是去調用Handler的invoke方法,其中參數是封裝say()的Method對象測試

JDK動態代理this

  • 只能代理實現了接口的類
  • 沒有實現接口,就不能代理

cglib動態代理設計

  • 針對類來實現代理
  • 對指定目標類產生一個子類,經過攔截因此的父類方法完成代理

如下是一個cglib的基本用法代理

public class BProxy implements MethodInterceptor{

	//代理類實例
	private Enhancer enhancer = new Enhancer();
	//獲取代理類
	public Object getProxy(Class<?> cls){
		//設置父類,即原類
		enhancer.setSuperclass(cls);
		//回調
		enhancer.setCallback(this);
		//建立代理類
		return enhancer.create();
	}
	
	/** 
	 * 攔截因此目標類方法的調用
	 * @param obj 目標類的實例
	 * @param m 目標方法的反射對象
	 * @param arg 目標方法的參數
	 * @param proxy 代理類的實例
	 */
	@Override
	public Object intercept(Object obj, Method m, Object[] arg, MethodProxy proxy) throws Throwable {
		
		System.out.println("\n=================");
		//代理類調用父類的方法,即原方法
		proxy.invoke(obj, arg);
		System.out.println("=================");
		
		return null;
	}

}

注:以上爲我的感悟,未讀源碼驗證。code

相關文章
相關標籤/搜索