美文网首页
设计模式(二)动态代理

设计模式(二)动态代理

作者: 六脉神见 | 来源:发表于2018-09-18 21:50 被阅读0次

    定义:动态代理其实还是一种代理,只不过实际的代理类对象是在运行时通过反射动态创建的,所以称之为动态代理.
    我们知道代理模式中,每对一个类增加代理,就需要新建一个静态的代理类,如,卖房就要为卖方建一个代理类,卖车也要建一个,卖书,卖水果,包子,油条...如此这般,如果你的项目里需要代理大量的类,比方说每个方法前后都需要记录开始时间,结束时间,如果用静态代理,势必每个类都需要建立一个代理类.
    这个时候动态代理就这么横空出世了...
    先上UML:


    image.png

    模式中角色如下:
    1、核心类-Student,实际的业务逻辑放在该类,代理类也是代理这个类的行为
    2、核心类的抽象层接口-Person,该接口也就是和静态代理模式里一样的接口,用来规范核心类和代理类共同的方法
    3、代理类-Proxy,在动态代理中,代理类的实例其实是通过反射动态生成的
    4、InvocationHandler接口,该接口由JDK提供,使用动态代理需要实现该接口并覆写invoke()方法
    5、StudentInvocationHandler类,接口InvocationHandler的实现类
    说明:在客户调用该模式时,其事件流是Client->Proxy->StudentInvocationHandler->Student

    代码如下:

    /**
     * 接口类
     * @author saisaimayi
     *
     */
    public interface Person {
    
        public void payFees();
    }
    
    /**
     * 核心类,包含业务逻辑
     * @author saisaimayi
     *
     */
    public class Student implements Person {
    
        public void payFees() {
            System.out.println("上交班费...");
        }
    
    }
    
    /**
     * InvocationHandler的实现类,
     * @author saisaimayi
     *
     */
    public class StudentInvocationHandler implements InvocationHandler {
        
        private Person person;
        
        public StudentInvocationHandler(Person person){
            this.person = person;//通过构造方法将被代理的对象传进来
        }
    
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            beforePay();
            method.invoke(person, args);
            afterPay();
            return null;
        }
    
        private void afterPay() {
            System.out.println("缴费之后...");
        }
    
        private void beforePay() {
            System.out.println("缴费之前...");
        }
    
    }
    
    

    模式里的结构就只有上面三个了,说好的代理类呢?哪去了?
    别急,动态代理嘛,代理当然是动态生成了,如下,在客户调用的时候生成Proxy对象并调用:

    public class DynamicProxyTest extends TestCase {
        public void test(){
            //实例化核心业务类-就是被代理的那货
            Person liLei = new Student();       
            /**生成代理对象-最关键的地方
             * 三个参数,第一个核心类对应的加载器
             * 第二个是对应的接口列表
             * 第三个就是我们刚刚定义的Handler的一个实例化对象,注意需要把核心类                      
             * 的对象塞进去,这样才能实现从Handler->Student的传递
             * **/      
            Person proxy = (Person)Proxy.newProxyInstance(liLei.getClass().getClassLoader(), 
                    liLei.getClass().getInterfaces(), 
                    new StudentInvocationHandler(liLei));
            proxy.payFees();
        }
    }
    
    执行结果:
    缴费之前...
    上交班费...
    缴费之后...
    

    总结一下,从代码中可以看出,动态代理免去了我们创建大量类的尴尬,由此,我们想要代理谁,只要在代码中实例化代理对象就OK了。

    相关文章

      网友评论

          本文标题:设计模式(二)动态代理

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