欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

动态代理模式

程序员文章站 2022-03-09 13:33:01
...

2.动态代理类

        Java动态代理类位于Java.lang.reflect包下,一般主要涉及到以下两个类:

        (1). Interface InvocationHandler:该接口中仅定义了一个方法Objectinvoke(Object obj,Method method, Object[] args)。在实际使用时,第一个参数obj一般是指代理类,method是被代理的方法,如上例中的request()args为该方法的参数数组。 这个抽象方法在代理类中动态实现。
        (2).Proxy
:该类即为动态代理类,作用类似于上例中的ProxySubject,其中主要包含以下内容:

Protected Proxy(InvocationHandler h):构造函数,估计用于给内部的h赋值。

Static Class getProxyClass (ClassLoader loader, Class[] interfaces):获得一个代理类,其中loader是类装载器,interfaces是真实类所拥有的全部接口的数组。

       Static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h):返回代理类的一个实例,返回后的代理类可以当作被代理类使用(可使用被代理类的在Subject接口中声明过的方法)

     所谓Dynamic Proxy是这样一种class:它是在运行时生成的class,在生成它时你必须提供一组interface给它,然后该class就宣称它实现了这些 interface。你当然可以把该class的实例当作这些interface中的任何一个来用。当然啦,这个Dynamic Proxy其实就是一个Proxy,它不会替你作实质性的工作,在生成它的实例时你必须提供一个handler,由它接管实际的工作。

    在使用动态代理类时,我们必须实现InvocationHandler接口,以第一节中的示例为例:

      通过这种方式,被代理的对象(RealSubject)可以在运行时动态改变,需要控制的接口(Subject接口)可以在运行时改变,控制的方式(DynamicSubject)也可以动态改变,从而实现了非常灵活的动态代理关系。

/**
 * 
 * @功能:代理模式中公共的行为接口
 * @创建人 gao_jie
 * @创建日期 May 8, 2009
 * @版本 1.0
 * 
 */
public interface Subject {
	public abstract void request();
}

 

/**
 * 
 * @功能:代理模式中真正的实现
 * @创建人 gao_jie
 * @创建日期 May 8, 2009
 * @版本 1.0
 * 
 */
public class RealSubject implements Subject {	
	/**
	 * 构造函数
	 */
	public RealSubject(){	
	}
	/* (non-Javadoc)
	 * @see six.proxytrip.Subject#request()
	 */
	public void request() {
		System.out.println("this is a real request!");	
	}
}

 

/**
 * 
 * @功能:代理类
 * @创建人 gao_jie
 * @创建日期 May 8, 2009
 * @版本 1.0
 * 
 */
public class ProxySubject implements InvocationHandler {
	/**
	 * 声明对象
	 */
	private Object subject;
	/**
	 * 
	 */
	public ProxySubject() {
	}
	public ProxySubject(Object obj) {
		this.subject = obj;
	}
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		System.out.println( " before calling  "   +  method);
		method.invoke(subject,args);
		System.out.println( " after calling  "   +  method);
		return   null ;
	}
}

 

/**
 * 
 * @功 能 : 客户端测试代码
 * @创建人 gao_jie
 * @创建日期 May 8, 2009
 * @版本 1.0
 * 
 */
public class Client {
	/**
	 * @param args
	 */
	public static void main(String[] args) {
		 RealSubject rs  =   new  RealSubject();  // 在这里指定被代理类  
		   InvocationHandler ds  =   new  ProxySubject(rs); 
		   Class cls  =  rs.getClass(); 
		    // 以下是一次性生成代理 
		   Subject subject  =  (Subject) Proxy.newProxyInstance(cls.getClassLoader(), cls.getInterfaces(),ds ); 
		   subject.request(); 

	}
}

 

测试结果:
 before calling  public abstract void six.dydmicproxy.Subject.request()
this is a real request!
 after calling  public abstract void six.dydmicproxy.Subject.request()

     该代理类的内部属性为Object类,实际使用时通过该类的构造函数DynamicSubject(Object obj)对其赋值;此外,在该类还实现了invoke方法,该方法中的

method.invoke(sub,args);

其实就是调用被代理对象的将要被执行的方法,方法参数sub是实际的被代理对象,args为执行被代理对象相应操作所需的参数。通过动态代理类,我们可以在调用之前或之后执行一些相关操作。

相关标签: 工作