java 动态代理

         java代理可以分为三类:静态代理、JDK动态代理以及CGLIB动态代理。

       其中,像包装器模式就属于静态代理,具体参考相应的设计模式。另外,直接继承目标类,也可以实现静态代理。静态代理方式,代理类和目标对象耦合度比较高。

       java自带的动态代理就是JDK动态代理。主要通过Proxy代理类以及调用处理接口InvocationHandler来实现。下面重点说明下这种代理方式。

        >.1. Interface InvocationHandler:

           该接口是代理实例的调用处理程序需要实现的接口。每个代理实例都具有一个关联的调用处理程序。对代理实例调用方法时, 将对方法调用进行编码并将其指派到它的调用处理程序的invoke 方法。

           该接口下只包含一个方法, Object  invoke(Object proxy,Method method, Object[] args)

           其中,proxy---代理实例;

                      method---代理实例执行目标对象的方法实例;

                      args---执行方法的参数数组。

         >.2. Proxy 即代理类:

            提供了用于创建动态代理类和实例的静态方法。

                 主要方法:Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces, InvocationHandler h)

             该方法返回一个指定接口的代理类实例,

             其中,loader---定义代理类的类加载器;

                        interfaces---代理类实现的接口列表;

                        h---指派方法调用的调用处理程序。

             其他具体方法可以参考JDK文档。

            下面通过一个具体的Demo来演示动态代理的实现:

             首先,定义一个接口

       public interface ISubject
       {
 	   public void doWork();
       }

               然后,定义一个该接口的具体实现类

       public class ASubjectImpl implements ISubject
       {
	   @Override
	   public void doWork()
	   {
		System.out.println("A Subject do something!");		
	   }
       }

               接着,是一个代理实现类

       public class ProxyHandler implements InvocationHandler
       {
	   private Object proxied;
	
	   public ProxyHandler(Object proxied)
	   {
		this.proxied = proxied;
	   }

	   @Override
	   public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable
	   {
		// 在调用对象方法之前可以做一些工作
		//...
		//执行代理对象方法
		Object result = method.invoke(proxied, args);
		
		//执行代理对象方法之后执行一些操作
		return result;
	   }
       }

               最后,测试代码:

        public static void main(String args[])
	{
		//创建代理对象
		ISubject proxySubject = (ISubject) Proxy.newProxyInstance(ISubject.class.getClassLoader(), 
				new Class[]{ISubject.class}, new ProxyHandler(new ASubjectImpl()));
		
		//执行代理操作
		proxySubject.doWork();
	}

              执行结果很明显。上面代码的执行过程如下所示:

           

               对目标对象的调用,被Proxy拦截,并在InvocationHandler的回调方法中通过反射调用目标对象。

           Java自带的动态代理的限制在于只能对实现了接口的目标类进行代理,即接口代理。要想对类进行代理,可以通过CGLIB来实现。CGLIB采用生成目标类的子类的技术创建代理对象,所以不能对目标类的finally方法或者finally类进行代理。

           另外,CGLIB底层直接采用ASM编码方式,因此创建的代理对象性能更高。但是,创建代理对象的过程还是JDK动态代理性能更好。因此,对于单例对象来说,只创建一次,可以采用CGLIB动态代理,而非单例对象,因为每次都是创建新的对象,可以考虑代用JDK动态代理。

          Spring AOP的底层实现就是通过动态代理来支持的。下节会分析下Spring AOP的动态代理实现。



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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值