Java动态代理

代理,就是须要代理类和被代理类有同样的对外接口或者说成服务。所以代理类一般都必须实现了全部被代理类已实现的接口,由于接口就是制定了一系列对外服务的标准。

 

        正由于动态代理有这样灵活的特性。所以我们在设计动态代理类(DynamicProxy)时不用显式地让它实现与真实主题类(RealSubject)同样的接口(interface),而是把这样的实现推迟到执行时。

        为了能让DynamicProxy类可以在执行时才去实现RealSubject类已实现的一系列接口并执行接口中相关的方法操作。须要让DynamicProxy类实现JDK自带的java.lang.reflect.InvocationHandler接口。该接口中的invoke()方法可以让DynamicProxy实例在执行时调用被代理类的“对外服务”,即调用被代理类须要对外实现的全部接口中的方法,也就是完毕对真实方法的调用,Java帮助文档中称这些真实方法为处理程序。

        我们肯定必须先把被代理类RealSubject已实现的全部interface都载入到JVM中。不然JVM怎么可以找到这些方法呢?明确了这个道理。那么我们就行创建一个被代理类的实例,获得该实例的类载入器ClassLoader。

        所谓的类载入器ClassLoader,就是具有某个类的类定义,即类的内部相关结构(包含继承树、方法区等等)

        动态代理模式能够使得我们在不改变原来已有的代码结构的情况下,对原来的“真实方法”进行扩展、增强其功能,而且能够达到控制被代理对象的行为的目的。

DynamicProxy类必须实现的invoke()方法在调用被代理类的真实方法的前后都可进行一定的特殊操作。这是动态代理最明显的长处


为了解决某些问题,比方,不同意直接訪问某些类;对訪问要做特殊处理等。

或者,要对原方法进行统一的扩展。出现了动态代理需求

类图


JDK动态代理中包括一个类和一个接口: 
InvocationHandler接口: 

public interface InvocationHandler { 
public Object invoke(Object proxy,Method method,Object[] args) throws Throwable; 
} 


參数说明: 
Object proxy:指被代理的对象。

 
Method method:要调用的方法 
Object[] args:方法调用时所须要的參数 

能够将InvocationHandler接口的子类想象成一个代理的终于操作类

接口中声明的全部方法都被转移到一个集中的方法中处理(invoke)

动态代理类仅仅能代理接口,代理类都须要实现InvocationHandler类,实现invoke方法。该invoke方法就是调用被代理接口的全部方法时须要调用的,该invoke方法返回的值是被代理接口的一个实现类

主题接口

public interface ISubject {
	void doSomething();
}

真实对象

public class RealSubject implements ISubject {
	@Override
	public void doSomething() {
		System.out.println("+++RealSubject+++");
	}
}

handler

public class SubHandler implements InvocationHandler {
	private Object obj;// 持有真实对象
	public SubHandler(Object obj) {
		this.obj = obj;
	}
	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		Object result = method.invoke(obj, args);// 真实对象来运行详细的方法
		System.out.println("dy proxy ");//增加自己操作
		return result;
	}
}

使用

public class Test {
	public static void main(String[] args) {
		RealSubject realSubject = new RealSubject();
		ISubject sub = (ISubject) Proxy.newProxyInstance(realSubject.getClass()
				.getClassLoader(), realSubject.getClass().getInterfaces(),
				new SubHandler(realSubject));
		sub.doSomething();
	}
}

//output

+++RealSubject+++

dy proxy






posted @ 2017-05-23 10:09  yangykaifa  阅读(119)  评论(0编辑  收藏  举报