代理模式详细介绍

     这段时间在学习spring的AOP,而AOP的基础就是动态代理,所以本文详细介绍代理模式(包括动态代理和静态代理)

代理模式:

     代理模式的作用:为其他对象提供一种代理以控制对这个对象的访问。

     在某些情况下,一个客户不想或者不能直接引用一个对象,而代理对象可以在客户端和目标对象之间起到中介作用。

静态代理:

1、首先抽象角色类,可以定义一个接口或者抽象类:

<pre name="code" class="java">
/**
 * 真实对象和代理对象共同的接口
 *
 */
public interface Subject
{
	public abstract void request();
}
 

2、真实对象,完成实际工作:

/**
 * 真实角色
 *
 */
public class RealSubject implements Subject
{

	public void request()
	{
		System.out.println("I am realSubejct");
	}
	
}
3、代理对象,内部包含对真实对象的引用,并且可以附加自己的一些操作:
/**
 * 代理角色
 *
 */
public class ProxySubejct implements Subject
{
	//代理对象中含有对真实对象的引用
	private RealSubject realSubject;
	
	public void request()
	{
		//调用真实对象之前的操作
		System.out.println("preRequest");
		
		if(null == realSubject)
		{
			realSubject = new RealSubject();
		}
		//真实对象完成的动作
		realSubject.request();
		
		//调用真实对象后完成的附加操作
		System.out.println("post");
	}
}

4、客户端的调用:

public class Test
{
	public static void main(String[] args)
	{
		Subject subject = new ProxySubejct();
		subject.request();
	}
}

</pre><span style="font-family: SimSun; font-size: 18px;">5、打印结果:</span><p></p><p></p><pre name="code" class="java"><pre name="code" class="java">preRequest
I am realSubejct
post


 
    如上展示,静态代理可以实现客户端通过调用代理对象,从而间接调用了真实对象,并且在调用真实对象的同时还可以附件另外一些操作。但静态的代理的缺点是真实对象必须已经存在,且作为代理对象的内部属性,这就意味着一个真实对象必须对应一个代理对象,这样在对象多的情况下会导致类的急剧膨胀;另外如果事先不知道真实角色,该如何使用代理呢?为了解决这两个问题,下面介绍动态代理。 


动态代理:

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


Java的动态代理类位于java.lang.reflect包下,一般会涉及到两个类:

Proxy:代理类

InvocationHandler:

该接口中仅定义一个方法,      

<pre name="code" class="java">public Object invoke(Object proxy, Method method, Object[] args)
 

 其中,proxy代指代理类,method指的是被代理的方法,args指的是该方法的参数数组(无参时设置为null)

这个抽象方法在代理类中动态实现。


1、首先定义抽象角色和真实角色:  

<pre name="code" class="java">/**
 * 抽象角色
 *
 */
public interface Subject
{
	public abstract void request();
}


 

<pre name="code" class="java">/**
 * 真实角色
 *
 */
public class RealSubject implements Subject
{

	public void request()
	{
		System.out.println("I am realSubejct");
	}
	
}


 

2、之后定义一个DynamicSubject类:    

<pre name="code" class="java">/**
 * 该代理类的内部属性是一个object,实际使用的时候可以通过构造器传递一个对象
 * 该类实现了invoke方法,该方法中的method.invoke()本质是调用被代理类对象将要执行的方法
 * 通过动态代理类,我们可以在执行真实对象方法的前后加入自己额外的方法
 *
 */
public class DynamicSubject implements InvocationHandler
{
	//对真实对象的引用
	private Object obj;
	
	public DynamicSubject(Object obj)
	{
		this.obj = obj;
	}

	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable
	{
		System.out.println("Before calling:" + method);
		
		//通过反射调用方法
		method.invoke(obj,args);
		
		System.out.println("After calling:" + method);
		
		return null;
	}

}


 


3、客户端调用:

<pre name="code" class="java">public class Client
{
	public static void main(String[] args)
	{
		RealSubject realSubject = new RealSubject();
		InvocationHandler handler = new DynamicSubject(realSubject);
		
		//动态生成代理类,代理类转换成了接口类型
		Subject subject = (Subject) Proxy.newProxyInstance(realSubject
				.getClass().getClassLoader(), realSubject.getClass()
				.getInterfaces(), handler);
		//调用代理类的request,实际上是由handler接管,调用handler的invoke()方法
		subject.request();
		
		System.out.println(subject.getClass());
	}
}


 


4、运行的结果:

<pre name="code" class="java">Before calling:public abstract void DynamicProxy.Subject.request()
I am realSubejct
After calling:public abstract void DynamicProxy.Subject.request()
class com.sun.proxy.$Proxy0


 
其中$Proxy0为动态生成的代理类的名称。 

动态代理模式最关键的地方就是调用代理类的request方法时,为什么会跳转到调用handler对象的invoker()方法,下面我把动态生成的代理类输出并反编译出来。

5、生成动态代理类的方法:

<pre name="code" class="java">	/**
	 * 将动态代理类输出到文件中 
	 */
	public static void createProxy()
	{
		String name = "proxy";
		byte[] data = ProxyGenerator.generateProxyClass(name,new Class[]{RealSubject.class});
		
		FileOutputStream fos;
		try
		{
			fos = new FileOutputStream(name + ".class");
			fos.write(data);
			fos.close();
		}
		catch (FileNotFoundException e)
		{
			e.printStackTrace();
		}
		catch (IOException e)
		{
			e.printStackTrace();
		}
	}


 
 
 

6、将生成的动态代理类proxy.class反编译,展示如下:

<pre name="code" class="java">import DynamicProxy.RealSubject;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException;

public final class proxy extends Proxy
  implements RealSubject
{
  private static Method m4;
  private static Method m1;
  private static Method m3;
  private static Method m7;
  private static Method m5;
  private static Method m9;
  private static Method m0;
  private static Method m8;
  private static Method m6;
  private static Method m2;

  public proxy(InvocationHandler paramInvocationHandler)
    throws 
  {
    super(paramInvocationHandler);
  }

  public final void wait(long paramLong)
    throws InterruptedException
  {
    try
    {
      this.h.invoke(this, m4, new Object[] { Long.valueOf(paramLong) });
      return;
    }
    catch (Error|RuntimeException|InterruptedException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final boolean equals(Object paramObject)
    throws 
  {
    try
    {
      return ((Boolean)this.h.invoke(this, m1, new Object[] { paramObject })).booleanValue();
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final void request()
    throws 
  {
    try
    {
      this.h.invoke(this, m3, null);
      return;
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final Class getClass()
    throws 
  {
    try
    {
      return (Class)this.h.invoke(this, m7, null);
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final void wait()
    throws InterruptedException
  {
    try
    {
      this.h.invoke(this, m5, null);
      return;
    }
    catch (Error|RuntimeException|InterruptedException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final void notifyAll()
    throws 
  {
    try
    {
      this.h.invoke(this, m9, null);
      return;
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final int hashCode()
    throws 
  {
    try
    {
      return ((Integer)this.h.invoke(this, m0, null)).intValue();
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final void notify()
    throws 
  {
    try
    {
      this.h.invoke(this, m8, null);
      return;
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final void wait(long paramLong, int paramInt)
    throws InterruptedException
  {
    try
    {
      this.h.invoke(this, m6, new Object[] { Long.valueOf(paramLong), Integer.valueOf(paramInt) });
      return;
    }
    catch (Error|RuntimeException|InterruptedException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final String toString()
    throws 
  {
    try
    {
      return (String)this.h.invoke(this, m2, null);
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  static
  {
    try
    {
      m4 = Class.forName("DynamicProxy.RealSubject").getMethod("wait", new Class[] { Long.TYPE });
      m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[] { Class.forName("java.lang.Object") });
      m3 = Class.forName("DynamicProxy.RealSubject").getMethod("request", new Class[0]);
      m7 = Class.forName("DynamicProxy.RealSubject").getMethod("getClass", new Class[0]);
      m5 = Class.forName("DynamicProxy.RealSubject").getMethod("wait", new Class[0]);
      m9 = Class.forName("DynamicProxy.RealSubject").getMethod("notifyAll", new Class[0]);
      m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
      m8 = Class.forName("DynamicProxy.RealSubject").getMethod("notify", new Class[0]);
      m6 = Class.forName("DynamicProxy.RealSubject").getMethod("wait", new Class[] { Long.TYPE, Integer.TYPE });
      m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
      return;
    }
    catch (NoSuchMethodException localNoSuchMethodException)
    {
      throw new NoSuchMethodError(localNoSuchMethodException.getMessage());
    }
    catch (ClassNotFoundException localClassNotFoundException)
    {
      throw new NoClassDefFoundError(localClassNotFoundException.getMessage());
    }
  }
}


 

    可以看出,动态代理类实现RealSubject接口,并继承Proxy。直到这里,我们已经很清楚地知道动态代理模式在调用代理类的request方法时,为什么会跳转到调用handler对象的invoker()方法。在反编译中,列出了proxy类所有的方法,其中在reqeust的方法中,就是通过handler对象调用invoker()方法。而handler对象就是通过动态代理类的父类Proxy的构造函数注入的。另外调用的request方法,就是通过反射获取的:

<pre name="code" class="java">m3 = Class.forName("DynamicProxy.RealSubject").getMethod("request", new Class[0]); 


 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值