美文网首页Java设计模式程序员
设计模式-动态代理模式

设计模式-动态代理模式

作者: monkey01 | 来源:发表于2017-09-23 21:16 被阅读0次

    之前介绍了代理模式,大家也都了解了代理模式,不过之前介绍的代理模式是静态代理,静态代理什么意思?静态代理指的是代理类是事先定义好的,在使用代理类的时候也是明确初始化定义的代理类。但是如果有非常多的类需要代理,那么一个个那么定义静态代理类的工作量将是非常大的,其实JDK已经给我们提供了现成的动态代理类,可以不用事先定义好所有的代理类,下面我们就介绍下JDK动态代理类。

    JDK中动态代理主要涉及到的是两个类:java.lang.reflect.InvocationHandler和java.lang.reflect.Proxy。下面我们先定义一下Subject接口和两个功能一样的实现类用于测试。

    Subject.java 接口类

    public interface Subject {
        public String doSomething(String name);
    }
    

    RealSubject.java 实际提供功能的类

    public class RealSubject implements Subject {
        @Override
        public String doSomething(String name) {
            System.out.println(name + " do something!");
            return name + " do something!";
        }
    }
    

    需要实现动态代理功能的话实现一个代理类实现InvocationHandler的接口,并实现invoke方法。

    例子中实现DynamicProxyAOP类用于在Subject类前后分别实现其他功能,类似AOP切面功能,其实Spring AOP的内部核心实现就是通过动态代理实现的。可以看到public Object invoke(Object proxy, Method method, Object[] args)方法有3个入参,proxy表示代理类,method表示调用的Subject主类的方法名,args表示调用代理方法的参数数组,在内部的具体实现我们通过反射方法通过method.invode来调用实际Subject方法。

    
    public class DynamicProxyAOP implements InvocationHandler {
        private Object subject;
    
        public DynamicProxyAOP(Object subject) {
            this.subject = subject;
        }
    
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            //在执行真实subject执行的方法
            System.out.println("before do something");
            //执行真实subject方法
            Object rtn = method.invoke(subject, args);
            //在执行结束后再执行的方法
            System.out.println("after do something");
    
            return rtn;
        }
    }
    

    这样就定义好了动态代理的核心类,最后我们看下如何调用动态代理类。先分别定义RealSubject和RealSubject2两个类的对象,再分别实现两个DynamicProxyAOP的代理对象,DynamicProxyAOP的入参分别为这两个Subject的对象。接下来我们创建动态代理对象,通过调用Proxy的Object newProxyInstance(ClassLoader loader,
    Class<?>[] interfaces,
    InvocationHandler h)来创建动态代理对象,第一个参数为实现代理类的ClassLoader,第二个参数为被代理类的接口数组,最后一个参数为继承自InvocationHandler的handler实现具体的代理功能。

    public class DynamicProxyMain {
    
        public static void main(String[] args){
            Subject realSubject = new RealSubject();
            Subject realSubject2 = new RealSubject2();
    
            InvocationHandler handler = new DynamicProxyAOP(realSubject);
            InvocationHandler handler2 = new DynamicProxyAOP(realSubject2);
    
            Subject subject = (Subject) Proxy.newProxyInstance(realSubject.getClass().getClassLoader(),
                    realSubject.getClass().getInterfaces(),
                    handler);
    
            Subject subject2 = (Subject) Proxy.newProxyInstance(realSubject2.getClass().getClassLoader(),
                    realSubject2.getClass().getInterfaces(),
                    handler2);
    
            String rtn = subject.doSomething("tester ");
            String rtn2 = subject2.doSomething("tester ");
        }
    }
    

    从打印出的日志可以发现,定义了两个不同的继承自Subject的类,但是动态代理对象是动态创建的,不需要像静态代理那样,再为每一个Subject实现类创建代理类,是不是很方便,后续大家如果要自己实现一些切面功能,就可以通过动态代理模式来实现,例如纪录日至之类的。

    
    before do something
    tester  do something!
    after do something
    before do something
    tester  do something2!
    after do something
    

    本文实例中的源码:https://github.com/feiweiwei/designpattern

    相关文章

      网友评论

        本文标题:设计模式-动态代理模式

        本文链接:https://www.haomeiwen.com/subject/rfchextx.html