技术成神之路:设计模式(十六)代理模式

介绍

代理模式(Proxy Pattern)是一种结构性设计模式,它通过代理对象来控制对另一个对象的访问。代理对象在功能上与真实对象相似,但可以在访问真实对象前后添加一些额外的处理。代理模式常用于控制对某个对象的访问、延迟实例化、权限控制等场景。

1.定义


通过引入一个代理对象(Proxy)来控制对真实对象(RealSubject)的访问。代理对象与真实对象实现相同的接口,通过代理对象的接口调用,客户端可以间接地与真实对象进行交互。

2. 主要作用


  • 控制对象访问:通过代理对象控制对目标对象的访问权限。
  • 延迟实例化:通过虚拟代理,可以在真正需要对象时再创建对象,实现延迟加载(Lazy Loading)。
  • 增强功能:在访问对象时,可以通过代理类来添加一些额外的功能,例如日志记录、性能统计等。
  • 远程访问:远程代理可以让客户端访问在不同地址空间的对象(如远程服务器上的对象)。

3. 解决的问题


  • 避免在不必要时创建昂贵的对象实例。
  • 提供一种简单的方式来管理对真实对象的访问权限。
  • 通过代理层降低客户端与真实对象的耦合度。

4. 模式原理


包含角色:

  1. 主题接口(Subject):真实对象和代理对象都实现该接口,以确保它们可以互换。
  2. 真实主题(RealSubject):实现了主题接口,包含具体的业务逻辑。
  3. 代理(Proxy):实现了主题接口,持有对真实主题的引用,控制对其的访问。

UML类图:

简单示例:

java 复制代码
//定义接口 Subject
public interface Subject {
    void request();
}

//实现具体的 RealSubject 类
public class RealSubject implements Subject {
    @Override
    public void request() {
        System.out.println("RealSubject:处理请求。");
    }
}

//实现 Proxy 类
public class Proxy implements Subject {
    private RealSubject realSubject;

    @Override
    public void request() {
        if (realSubject == null) {
            realSubject = new RealSubject();
        }
        System.out.println("代理:在请求传递给真正的主体之前记录请求。");
        realSubject.request();
    }
}

调用

java 复制代码
public class Client {
    public static void main(String[] args) {
        Subject proxy = new Proxy();
        proxy.request();
    }
}

打印输出

复制代码
代理:在请求传递给真正的主体之前记录请求。
RealSubject:处理请求。

以上一个简单的示例 实现了 代理实例化延迟加载,可以看到代理模式是如何通过代理对象控制对真实对象的访问。

代理模式可以说在我们的开发中非常常见,如果你看过三方库的源码的话,你就会发现几乎所有优秀的三方库都使用了代理模式,以安卓为例,

常见的有:
Retrofit: 它通过接口定义API服务,并使用动态代理来实现接口方法。当调用接口方法时,Retrofit会生成一个代理对象,该对象负责构建和发送HTTP请求。

EventBus: 利用代理模式来处理事件的订阅和发布。通过注解或手动注册,EventBus可以将事件处理逻辑与事件的发布者解耦。

还有 OkHttpGlideDagger等等... 都合理地使用了代理模式。

因为模式结构本身不复杂,在实际开发中更容易被我们接纳,比如设计一个网络请求框架,我们只需封装好 网络请求框架,也就是抽象接口,在这里叫主题接口 ,至于getpost ...请求的具体实现交给实现类去做,我们不需要考虑使用什么网络框架,假如我们使用的是OkHttp,哪怕有一天OkHttp不能使用了,我们也只需新增个真实主题使用其他网络库,然后一行代码替换,而不用在每个网络请求的地方都去修改代码,这就是代理模式的魅力。

5. 优缺点


优点:

  1. 透明性:客户端无需知道代理的存在,仍然可以以一致的方式访问真实对象。
  2. 灵活性:可以在不修改客户端代码的情况下,轻松替换真实对象。
  3. 职责分离:将真实对象的复杂性与代理的控制逻辑分开,便于管理和维护。

缺点:

  1. 增加复杂性:增加了系统的复杂性,尤其是代理层次较多时。
  2. 性能开销:代理模式会引入额外的调用开销,尤其是远程代理。

凡事有利也有弊,我认为代理模式的优点更为突出,至于缺点,不能忽略,但是在那些大型项目上都在使用,也几乎可以忽略了。

6. 应用场景


  • 当对象的创建开销较大且不常用时,可以使用虚拟代理延迟创建。
  • 需要访问远程服务器上的对象时。
  • 需要对对象的访问进行控制和权限管理时。
  • 在访问对象时添加一些额外的操作,如引用计数、日志记录等。
  • ...

7. 总结


代理模式通过引入代理对象,为访问真实对象提供了一种灵活的控制手段。它不仅能够优化资源使用,还能增强系统的安全性和可维护性。在设计系统时,合理地应用代理模式,可以有效地解决访问控制、性能优化等问题。然而,在复杂性和性能开销之间需进行权衡,以避免不必要的复杂实现。

代理模式是日常开发中非常常见的设计模式之一,理解并熟练运用它可以大大提高代码的灵活性和可维护性。

相关推荐
WispX8881 小时前
【设计模式】门面/外观模式
java·开发语言·设计模式·系统架构·外观模式·插件·架构设计
蔡蓝1 小时前
设计模式-外观模式
microsoft·设计模式·外观模式
琢磨先生David1 小时前
简化复杂系统的优雅之道:深入解析 Java 外观模式
java·设计模式·外观模式
on the way 12314 小时前
结构性设计模式之Flyweight(享元)
java·设计模式·享元模式
暴躁哥18 小时前
深入理解设计模式之访问者模式
设计模式·访问者模式
佩奇的技术笔记18 小时前
从Java的JDK源码中学设计模式之装饰器模式
java·设计模式·装饰器模式
on the way 12318 小时前
结构型设计模式之Proxy(代理)
设计模式·代理模式
YGGP21 小时前
【结构型模式】装饰器模式
设计模式
将编程培养成爱好1 天前
《复制粘贴的奇迹:小明的原型工厂》
c++·设计模式·原型模式
liang_jy1 天前
设计模式中的几大原则
设计模式·面试