您的位置:首页 > 编程语言 > Java开发

java 动态代理

2015-08-11 15:12 459 查看
什么是动态代理:

说起动态,其实不如先说什么是静态。所谓静态代理,个人理解为自己手写的代理类,或者用工具生成的代理类,或者别人帮你写的代理类(没说一样...)。总之,就是程序运行前就已经存在的编译好的代理类。

相反,如果代理类程序运行前并不存在,需要在程序运行时动态生成(无需手工编写代理类源码),那就是今天要说的动态代理了。

如何生成的:根据Java的反射机制动态生成。

不多说了,上程序。

目标接口TargetInterface:

Java代码  


public interface TargetInterface {  

    public int targetMethodA(int number);  

    public int targetMethodB(int number);  

}  

很简单,一个普通的接口,里面有若干方法(此处写2个示范一下)

实现该接口的委托类ConcreteClass:

Java代码  


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;  

    }  

  

}  

很简单,一个普通的类,实现了目标接口。

代理处理器类ProxyHandler:

Java代码  


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;  

    }  

  

}  

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

最后是入口程序:

Java代码  


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);  

    }  

}  

首先创建委托类对象,将其以构造函数传入代理处理器,代理处理器ProxyHandler中会以Java反射方式调用该委托类对应的方法。然后使用Java反射机制中的Proxy.newProxyInstance方式创建一个代理类实例,创建该实例需要指定该实例的类加载器,需要实现的接口(即目标接口),以及处理代理实例接口调用的处理器。

最后,调用代理类目标接口方法时,会自动将其转发到代理处理器中的invoke方法内,invoke方法内部实现预处理,对委托类方法调用,事后处理等逻辑。

 

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

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

2、系统扩展性和维护性增强,程序修改起来也方便多了(一般只要改代理处理器类就行了)。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  java 动态代理