Java动态代理学习

一、动态代理VS静态代理

静态代理:就是程序运行前就已经存在的编译好的代理类。设计模式中的代理模式编写的代理类即为静态代理。
动态代理:代理类程序运行前并不存在,需要在程序运行时动态生成(无需手工编写代理类源码),那就是今天要说的动态代理了。根据Java的反射机制动态生成。

二、动态代理标准实现流程

1、实现目标接口

public interface TargetInterface {  
    public int targetMethodA(int number);  
    public int targetMethodB(int number);  
} 

2、实现该接口的委托类ConcreteClass:

 public class ConcreteClass implements TargetInterface{  

        public int targetMethodA(int number) {  
            System.out.println("开始调用目标类的方法targetMethodA...");  
            System.out.println("操作-打印数字:"+number);  
            System.out.println("结束调用目标类的方法targetMethodA...");  
            return number;  
        }  

        public int targetMethodB(int number){  
            System.out.println("开始调用目标类的方法targetMethodB...");  
            System.out.println("操作-打印数字:"+number);  
            System.out.println("结束调用目标类的方法targetMethodB...");  
            return number;  
        }  
    }  

3、代理处理器类ProxyHandler,必须实现InvocationHandler 接口

该类实现了Java反射包中的InvocationHandler接口。代理实例调用方法时,将对方法调用指派到它的代理处理器程序的invoke方法中。invoke方法内部实现预处理,对委托类方法调用,事后处理等逻辑。

public class ProxyHandler implements InvocationHandler{  
    private Object concreteClass;  

    public ProxyHandler(Object concreteClass){  
        this.concreteClass=concreteClass;  
    }  

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {  
        System.out.println("proxy:"+proxy.getClass().getName());  
        System.out.println("method:"+method.getName());  
        System.out.println("args:"+args[0].getClass().getName());  

        System.out.println("Before invoke method...");  
        Object object=method.invoke(concreteClass, args);//普通的Java反射代码,通过反射执行某个类的某方法  
        //System.out.println(((ConcreteClass)concreteClass).targetMethod(10)+(Integer)args[0]);  
        System.out.println("After invoke method...");  
        return object;  
    }  
} 

4、程序入口

public class DynamicProxyExample {  
    public static void main(String[] args){  
         ConcreteClass c=new ConcreteClass();//元对象(被代理对象)  
         InvocationHandler ih=new ProxyHandler(c);//代理实例的调用处理程序。  
         //创建一个实现业务接口的代理类,用于访问业务类(见代理模式)。  
         //返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序,如ProxyHandler。  
         TargetInterface targetInterface=  
             (TargetInterface)Proxy.newProxyInstance(c.getClass().getClassLoader(),c.getClass().getInterfaces(),ih);  
         //调用代理类方法,Java执行InvocationHandler接口的方法.  
         int i=targetInterface.targetMethodA(5);  
         System.out.println(i);  
         System.out.println();  
         int j=targetInterface.targetMethodB(15);  
         System.out.println(j);  
    }  
}

5、写在后面

使用Java动态代理机制的好处:

1、减少编程的工作量:假如需要实现多种代理处理逻辑,只要写多个代理处理器就可以了,无需每种方式都写一个代理类。

2、系统扩展性和维护性增强,程序修改起来也方便多了(一般只要改代理处理器类就行了)。

使用Java动态代理机制的限制:

目前根据GOF的代理模式,代理类和委托类需要都实现同一个接口。也就是说只有实现了某个接口的类可以使用Java动态代理机制。但是,事实上使用中并不是遇到的所有类都会给你实现一个接口。因此,对于没有实现接口的类,目前无法使用该机制。有人说这不是废话吗,本来Proxy模式定义的就是委托类要实现接口的啊!但是没有实现接口的类,该如何实现动态代理呢?

当然不是没有办法,这也是我后面抽时间要继续整理和总结原先使用过的一件神器,相关Blog会不定期发上来。那就是大名鼎鼎的CGLib…

PS:CGLib中的动态代理已经新鲜出炉,欢迎访问:http://shensy.iteye.com/blog/1873155

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值