代理模式——七种结构型模式之一

作者: lanceJin | 来源:发表于2017-05-09 10:53 被阅读0次

    1.前言


    前面讲的模式提供了一种在创建对象的同时隐藏创建逻辑的方式,而不是直接实例化对象。但结构型模式关注的是如何组合类和对象,来获取作用更广泛的结构或者新的功能。
      结构型模式分为类模式和对象模式。前者通过继承类和实现接口的方式,使目标类具备所有父类和接口的性质;后者则将一些已经存在的对象组合起来,扩展了功能,同时可以动态改变组合关系,具有很大的灵活性。

    2.概念


    代理模式为其它对象提供一种代理以控制对这个对象的访问。当无法或不想直接访问某个对象时,通过一个代理对象来间接访问。根据适用范围分为四类(引用自《Android源码设计模式解析与实战》):

    • 远程代理:为某个对象在不同的内存地址空间提供局部代理。
    • 虚拟代理:代理一个十分消耗资源的对象,并在真正需要时才创建此对象。
    • 保护代理:使用代理控制对原始对象的访问。
    • 智能引用:在访问原始对象时执行一些自己的附加操作。

    3.场景


    有张三、李四两人分别想要租房和买房,但是他们都太忙了,共同约了一个中介全权处理。需要注意的是,租房有租房的流程,买房有买房的手续。

    4.写法


    租房的房源较多,中介决定先帮张三处理好。

    // 1.声明需要代理的操作
    public interface Tenant {
        
        void rent();
    
    }
    
    // 2.被代理者实现操作
    public class ZhangSan implements Tenant {
    
        @Override
        public void rent() {
            System.out.println("我要租这套房");
        }
    
    }
    
    // 3.代理者调用被代理者的方法
    public class Middleman implements Tenant {
        
        private Tenant customer;
    
        public Middleman(Tenant customer) {
            super();
            this.customer = customer;
        }
    
        @Override
        public void rent() {
            customer.rent();
        }
    
    }
    
    public class Main {
    
        public static void main(String[] args) {
            // 租房子由中介出面
            Tenant zhangSan = new ZhangSan();
            Tenant middleMan = new Middleman(zhangSan);
            middleMan.rent();
        }
    
    }
    

    中介在替张三找房时,又发现了一个非常适合李四的房源在出售。但是他现在走的是租凭流程,想去办理买房手续不全,显得很难办。

    public interface Buyer {
        
        void buy();
    
    }
    
    public class LiSi implements Buyer {
    
        @Override
        public void buy() {
            System.out.println("我要买这套房");
        }
    
    }
    

    虽然也可以让Middleman类实现Buyer接口获取买房的方法,但是代理者同时具有两个被代理者的方法,容易产生混淆,导致代理过程不透明,不符合代理模式设计的愿望。
      上面这种方式,由于代码运行前,代理者的.class编译文件已存在,即代理关系已确定,所以称之为静态代理。如果不声明代理者就可以不确定代理关系,从而在执行阶段决定代理谁,需要用到Java的反射机制动态地生成代理者对象,这便是动态代理。与原型模式类似,Java也提供了一个动态代理接口InvocationHandler,通过重写 invoke() 方法实现动态调用被代理者的方法。

    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    
    public class DynamicProxy implements InvocationHandler {
        
        private Object customer;
    
        public DynamicProxy(Object customer) {
            super();
            this.customer = customer;
        }
    
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            // 1.通过反射调用被代理对象的相应方法
            Object result = method.invoke(customer, args);
            return result;
        }
    
    }
    
    import java.lang.reflect.Proxy;
    
    public class Main {
    
        public static void main(String[] args) {
            Tenant zhangSan = new ZhangSan();
            Buyer liSi = new LiSi();
            // 分别获得两人的全权代理
            DynamicProxy tenant = new DynamicProxy(zhangSan);
            DynamicProxy buyer = new DynamicProxy(liSi);
            // 2.通过ClassLoader加载业务接口和代理对象,生成新的代理对象
            Tenant zsProxy = (Tenant) Proxy.newProxyInstance(Tenant.class.getClassLoader(), new Class[] {Tenant.class}, tenant);
            zsProxy.rent();
            Buyer lsProxy = (Buyer) Proxy.newProxyInstance(Buyer.class.getClassLoader(), new Class[] {Buyer.class}, buyer);
            lsProxy.buy();
        }
    
    }
    

    虽然静态代理符合面向对象的原则,将对象关系声明清楚,但是不够灵活,需要声明多个代理对象。而动态代理,将代理者与被代理者解耦,使代理者不受被代理者变化影响,令模式结构清晰、简单。

    5.总结


    代理模式是之所以称为结构型模式,是因为这种结构为对象的访问增加了独立的处理空间,有利于添加相应的功能。就拿智能引用来说,将它优化一下,就是装饰模式了。

    相关文章

      网友评论

        本文标题:代理模式——七种结构型模式之一

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