java动态代理案例代码详解

java | 2019-09-13 10:02:39

1.java动态代理介绍:
动态代理可以提供对另一个对象的访问,同时隐藏实际对象的具体事实。代理一般会实现它所表示的实际对象的接口。代理可以访问实际对象,但是延迟实现实际对象的部分功能,实际对象实现系统的实际功能,代理对象对客户隐藏了实际对象。客户不知道它是与代理打交道还是与实际对象打交道。

2.使用Java动态代理机制的好处:
1)减少编程的工作量:假如需要实现多种代理处理逻辑,只要写多个代理处理器就可以了,无需每种方式都写一个代理类。
2)让你的程序代码易维护性,可扩展新更好,系统扩展性和维护性增强,程序修改起来也方便多了(一般只要改代理处理器类就行了)。
3)解耦,通过参数就可以判断真实类,不需要事先实例化,更加灵活多变。
4)可以让对应不上的接口,通过代理来对应上。


3.java动态代理案例代码:
目标接口TargetInterface:

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

很简单,一个普通的类,实现了目标接口。
代理处理器类ProxyHandler:

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方法内部实现预处理,对委托类方法调用,事后处理等逻辑。
最后是入口程序:

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方法内部实现预处理,对委托类方法调用,事后处理等逻辑。

登录后即可回复 登录 | 注册
    
关注编程学问公众号