动态代理

前两天看了网上的一篇文章:用jdk的动态代理实现AOP, 看的还不错,虽然那篇文章是启蒙性质的。

动态代理:
       Proxy模式在jive的设计中利用的很充分,jive中利用Proxy模式里进行权限的验证。缺点是对每个核心类都要写一个Proxy,重复代码的地方很多。动态代理的特点是,不要静态的写Proxy类,而是在运行时由jdk自己帮助生成类,然后注入到classloader中去。

代码:
public interface BusinessProcess{
          void processBusiness();
}

public class BusinessProcessImpl implements BusinessProcess{
          public void processBusiness(){
                System.out.println("in businessProcessImpl");
           }
}

如果要对BusinessProcessImpl增加权限验证功能,那么静态代理代码:
public class BusinessProcessProxy implements BusinessProcess{
          private BusinessProcess target;
          public BusinessProcessProxy(BusinessProcess target){
                   this.target  = target;
          }
          
          public void processBusiness(){
                 checkPermission();
                 this.target.processBusiness();
          }
}

动态代理:
public class PermissionHandler implements InvocationHandler {
    private Object target;
    public LoggerHandler(Object target){
        this.target = target;
    }

     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        checkPermission();
        Object result = method.invoke(this.target,args);
        return result;
    }
}

测试代码:
        BusinessProcessImpl impl = new BusinessProcessImpl();
        PermissionHandler handler = new PermissionHandler (impl);
        BusinessProcess proxy = (BusinessProcess)Proxy.newProxyInstance(impl.getClass().getClassLoader(),impl.getClass().getInterfaces(), handler);
        proxy.processBusiness();

        //proxy对象是BusinessProcess接口的一个实现类,但是不是BusinessProcessImpl的子类
        //!   (BusinessProcessImpl)proxy       错误!


看起来好像静态代理和动态代理没什么区别,但是仔细一看,静态代理代理的target只能是BusinessProcess对象,而动态代理却能够代理任何一个Object对象,这就是关键区别。所以程序中不需要为每个核心类写一个Proxy,而公用一个就可以。

动态代理的实现原理:
byte[] proxyClassFile = ProxyGenerator.generateProxyClass(proxyName, interfaces);
proxyClass = defineClass0(loader, proxyName,proxyClassFile, 0,proxyClassFile.length);
这是java.lang.reflect.Proxy的两行关键代码,第一行代码生成名为proxyName, 实现的接口为interfaces的一个类的字节码。
第二行代码,将生成的字节码注入到classloader中去。

第一次为某个object生成proxyClass可能比较耗费性能,但是Proxy方法作了缓存,可以弥补这一点。此外,动态代理只能够对interfaces进行动态的代理, 也就是说它先找出需要代理的对象实现的所有的interface, 然后只对所有的这些interface的所有的方法代理,对象除了interfaces以外的方法不予处理。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值