结构型设计模式 之 代理模式 (静态代理和动态代理)

前言

代理模式是一种结构型设计模式,主要用来解决直接访问对象带来的问题。目地是为其他对象提供一个代理以控制某个对象的访问。代理分为静态代理和动态代理,先说静态代理,概念总是太抽象,直接用代码说明吧。

静态代理类图如下:

结构型设计模式代理模式:静态代理类图.jpg

抽象主题 声明一个接口

// 抽象主题  是一个接口,以一个工厂接口为例
public interface IToolFactory(){
    public void makeProduct();
}

定义一个真实的主题,用来实实在在做事的,以工厂为例,用来加工产品

public class ARealFactory implement IToolFactory(){
    @Override
    public void makeProduct(){
        System.out.println("我加工生产出来了你们所需要的产品。");
    }
}

下面是一个最重要的角色,代理类Proxy 相当于代购者

public class Proxy implements IToolFactory(){
    ARealFactory factory;
    public Proxy(ARealFactory factory){
        this.factory = factory;
    }

    public void doBefore(){
        System.out.println("售前咨询,可以提出客户的需求");
    }
    public void doAfter(){
        System.out.println("售后服务,解决客户的后顾之忧");
    }

    @Override
    public void makeProduct(){
        if(factory == null){
          factory = new ARealFactory();
        }

        doBefore();
        factory.makeProduct();
        doAfter();
    }

}

我们的顾客需要购买产品,就需要找代购

public class Customer(){

      public static void main(String args[]){
              //实例化一个真实的工厂
           // IToolFactory factory = new ARealFactory();
            //代购者需要持有真实对象
            Proxy proxy = new Proxy();
            proxy.makeProduct();
      }
}

小结:

以上就是静态代理的设计模式的原理演示。静态代理的缺点就是违反了开闭原则,扩展能力差,可维护性差。因此就有了动态代理设计模式。

动态代理

在静态代理的基础上,我们进行改进:主要修改代理类Proxy, 通过反射获取动态代理对象。

public class MarkCompany implements InvokeHandler{
     private Object factory;
    public void setFactory(Object factory){
        this.factory = factory;
    }
    public Object getFactory(){
        return this.factory;
    }

    //  关键方法:获取动态代理对象
    public Object getProxyInstance(){
        return Proxy.newProxyInstance(factory.getclass().getClassLoader(),facory.getClass().getInterfaces(),this);
    }

  /**
     * 通过动态代理对象进行增强
     * @param proxy
     * @param method
     * @param args
     * @return
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy,Method method,Object[] args){
        
        doBefore();
        method.invoke(factory,args);
        doAfter();
        return null;
    }
    public void doBefore(){
        System.out.println("售前咨询,可以提出客户的需求");
    }
    public void doAfter(){
        System.out.println("售后服务,解决客户的后顾之忧");
    }


}

调用有点小小的区别:

IToolFactory  realFactory = new ARealFactory();
MarkCompany company = new MarkCompany();
company.setFactory(realFactory);

IToolFactory employ1 = (IToolCompany)company.getProxyInstance();
employ1.makeProduct();

结语

在实际开发过程中,由于静态代理的局限性,用到的地方较少,可以会用于代理对象较少,比较固定不需要扩展的场景。
虽然,动态代理运用到了反射,性能会不及静态代理,但是由于动态代理的各种好处优点,非常强大,牺牲点性能还是值得的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 执伞闲漫寂寥雨巷,细露沾罗裙,梧桐绕指,袅袅无绊,盈盈扇掩珊瑚唇。 氤氲霁雨,红妆凝眸而浅笑嫣然。烟雨江南执伞,入...
    9月满圆阅读 5,443评论 0 2
  • 当我重新翻起床头的那一本《朝花夕拾》,那涩黄色的纸张,伴随着字墨的香味,依然围绕在我的身旁。已经不知道...
    43吴天宇阅读 3,318评论 0 0
  • 学习有两种:一种是学习前人总结到书藉上面的知识,另一种是通过实践,学习怎样去发掘我们所不知晓的。早上好
    帅气的头脑阅读 2,299评论 0 0
  • 作业: 1.写一两段话(如回忆一个场景),其中包含有触觉和身体感觉。 早高峰的地铁站,犹如古战场,每个人脸上都是视...
    久迩阅读 3,613评论 2 0

友情链接更多精彩内容