Java专题设计模式篇

设计模式之(二)代理模式:模拟java范型中的代理模式

2020-10-31  本文已影响0人  千夜零一

引言

  因为要使用java中的范型,所以需要学会代理模式打基础;就比如要学习Retrofit网络框架,首先要学会建造者模式打基础。换句话说,Retrofit是设计模式之一:建造者模式的最佳试炼场;而同样java中的范型是代理模式的最佳试炼场。话不多说,静态、动态代理模式都要学习起来!


简介

  在代理模式(Proxy Pattern)中,一个类代表另一个类的功能。这种类型的设计模式属于结构型模式。
  在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口。


详细介绍

意图:为其他对象提供一种代理以控制对这个对象的访问。
主要解决:
  在直接访问对象时带来的问题,比如说:要访问的对象在远程的机器上。在面向对象系统中,有些对象由于某些原因(比如对象创建开销很大,或者某些操作需要安全控制,或者需要进程外的访问),直接访问会给使用者或者系统结构带来很多麻烦,我们可以在访问此对象时加上一个对此对象的访问层。
何时使用:想在访问一个类时做一些控制。
如何解决:增加中间层。
关键代码:实现与被代理类组合。
应用实例:
  1、Windows 里面的快捷方式。
  2、买房子或者租房子,这里的中介就是代理。
  3、买火车票不一定在火车站买,也可以去代售点。
   4、一张支票或银行存单是账户中资金的代理。支票在市场交易中用来代替现金,并提供对签发人账号上资金的控制。
  5、spring aop。
优点: 1、职责清晰。 2、高扩展性。 3、智能化。
缺点:
  1、由于在客户端和真实主题之间增加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢。
  2、实现代理模式需要额外的工作,有些代理模式的实现非常复杂。
使用场景:按职责来划分,通常有以下使用场景:
  1、远程代理。
  2、虚拟代理。
  3、Copy-on-Write 代理。
  4、保护(Protect or Access)代理。
  5、Cache代理。
  6、防火墙(Firewall)代理。
  7、同步化(Synchronization)代理。
  8、智能引用(Smart Reference)代理。
注意事项:
  1、和适配器模式的区别:适配器模式主要改变所考虑对象的接口,而代理模式不能改变所代理类的接口。
  2、和装饰器模式的区别:装饰器模式为了增强功能,而代理模式是为了加以控制。


实践一:静态代理

步骤1:创建服务类接口

/**
 * @data on 2020/10/30 4:28 PM
 * @auther armStrong
 * @describe 静态代理-创建服务类接口
 */
public interface BuyHouse {
      void buyHosue();
}

步骤2:创建接口实现类

/**
 * @data on 2020/10/30 4:29 PM
 * @auther 接口实现类
 * @describe 干买房这件事儿
 */
public class BuyHouseImpl implements BuyHouse {
    @Override
    public void buyHosue() {
        System.out.println("我要买房");
    }

步骤3:创建实现接口的静态代理类

/**
 * @data on 2020/10/30 4:31 PM
 * @auther armStrong
 * @describe 中介,做代理帮客户买房子(一条龙服务)
 */
public class BuyHouseProxy implements BuyHouse {

    public BuyHouse bHosue;

    public BuyHouseProxy(final BuyHouse buyhosue){
        this.bHosue = buyhosue;
    }

    @Override
    public void buyHosue() {
        System.out.println("买房前准备");
        bHosue.buyHosue();
        System.out.println("买房后装修");
    }
}

步骤4:模拟实现静态代理

public class Case60 extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_case60);
        //调用静态代理方法
        staticProxy();
    }

    //静态代理
    private void staticProxy() {
        //自己买房
        BuyHouse buyHouse = new BuyHouseImpl();
        buyHouse.buyHosue();
        //代理帮助买房
        BuyHouseProxy buyHouseProxy = new BuyHouseProxy(buyHouse);
        buyHouseProxy.buyHosue();
    }
}

实践二:动态代理

  代替实践一中的步骤3,不用创建静态代理类,创建处理器类交给JDK在运行时帮我们进行动态创建代理对象。

步骤3:创建动态代理处理器类

/**
 * @data on 2020/10/30 4:55 PM
 * @auther armStrong
 * @describe 动态代理处理器类
 */
public class DynamicProxyHandler implements InvocationHandler {

    private Object object;

    public DynamicProxyHandler(final Object object) {
        this.object = object;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("买房前准备");
        Object result = method.invoke(object, args);
        System.out.println("买房后装修");
        return result;
    }
}

步骤4:模拟实现动态代理

public class Case60 extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_case60);
        //调用实现静态代理
//        staticProxy();
        //调用实现动态代理jdk
        dynamicProxy();
        //CGLIB代理  //不写了
    }

    //动态代理(JDK方式)
    //无需手动创建代理类对象,真正的代理对象由JDK再运行时为我们动态的来创建。
    private void dynamicProxy() {
        BuyHouse buyHouse = new BuyHouseImpl();
        BuyHouse proxyBuyHouse = (BuyHouse) Proxy.newProxyInstance(BuyHouse.class.getClassLoader(), new Class[]{BuyHouse.class}, new DynamicProxyHandler(buyHouse));
        proxyBuyHouse.buyHosue();
    }
}

大功告成!

上一篇 下一篇

猜你喜欢

热点阅读