代理模式
代理模式
Bummon代理模式
一、定义
摘自百度百科: 所谓的代理者是指一个类别可以作为其它东西的接口。代理者可以作任何东西的接口:网上连接、存储器中的大对象、文件或其它昂贵或无法复制的资源。
二、角色分类
抽象主题角色(Subject)
定义了代理角色和真实主题角色的共同接口,代理角色通过该接口调用真实主题角色的方法。
真实主题角色(Real Subject)
实现了抽象主题角色的接口,代表真实的业务对象。
代理角色(Proxy)
实现了抽象主题角色定义的接口,并持有真实主题角色的引用,代表了真实主题角色的代理。在代理角色中,可以添加额外的功能,如记录日志、权限控制等,而这些功能并不是真实主题角色本身所具备的功能
客户(Client)
具体调用代理者的角色
三、UML图
四、实现方式
需求分析
假如我们有一个这样的场景:我们想要去看电影,但是我们嫌麻烦不想去电影院排队买票,这时候我们可以选择第三方的平台来买票,并且代理会为我们处理一切的购票相关的事务,包括选座、付款等。其实这种模式就相当于我们设计模式中的代理模式,其代理第三方平台,接下来我们用代码来实现一下这个业务。
具体实现
简单实现
抽象主题角色(Subject)
1 | public interface Subject { |
真实主题角色(Real Subject)
1 | public class RealSubject implements Subject { |
代理角色(Proxy)
1 | public class Proxy implements Subject { |
客户角色(Client)
1 | public class Client { |
运行结果
1 | Proxy:请求前执行逻辑 |
动态代理
我们有一个接口 Subject 和其一个实现类 RealSubject,我们要使用JDK的动态代理来生成一个代理对象Proxy,代理对象的调用会转发给RealSubject对象。
抽象主题角色(Subject)
1 | public interface Subject { |
真实主题角色(Real Subject)
1 | public class RealSubject implements Subject { |
动态代理角色(DynamicProxy)
1 | public class DynamicProxy implements InvocationHandler { |
客户角色(Client)
1 | public class Client { |
运行结果
1 | DynamicProxy: 请求前执行逻辑 |
在上述代码中,RealSubject
表示真实的业务实现类,DynamicProxy
表示动态代理类,Client
表示客户端。我们通过创建RealSubject
对象,然后创建一个代理对象proxy
来访问RealSubject
。
在代理对象中,我们实现了InvocationHandler
接口,并重写了其中的invoke()
方法,在调用代理对象的方法时,invoke()
方法会被自动调用。在invoke()
方法中,我们可以进行一些额外的操作,比如在调用实际对象的方法前后添加日志等。
使用Proxy.newProxyInstance()
方法来创建代理对象。该方法需要传入三个参数:ClassLoader
对象,Class
对象数组和InvocationHandler
对象。其中,ClassLoader
对象用于加载代理类,Class
对象数组表示代理类需要实现的接口列表,InvocationHandler
对象用于处理代理对象的方法调用。
五、应用场景
以下部分内容摘自 菜鸟教程
意图: 为其他对象提供一种代理,以控制对这个对象的访问。
主要解决: 解决了在直接访问对象时带来的问题,比如说:要访问的对象在远程的机器上。在面向对象系统中,有些对象由于某些原因(比如对象创建开销很大,或者某些操作需要安全控制,或者需要进程外的访问),直接访问会给使用者或者系统结构带来很多麻烦,我们可以在访问此对象时加上一个对此对象的访问层。
何时使用: 想在访问一个类时做一些控制。
如何解决: 增加中间层。
关键代码: 实现与被代理类组合。
应用实例:
- Windows中的快捷方式
- 猪八戒去找高翠兰,结果却是孙悟空变的,可以这样理解:把高翠兰的外貌抽象出来,高翠兰本人和孙悟空都实现了这个接口,猪八戒在访问高翠兰的时候看不出来这个是悟空,所以此时孙悟空是高翠兰的代理类
- 买火车票不一定在火车站买,也可以去代售点
- 一张支票或银行存单是账户中资金的代理。支票在市场交易中用来替代现金,并提供对签发账号上资金的控制
- Spring AOP
适用场景:
按职责划分通常有以下场景:
- 远程代理
- 虚拟代理
- Copy-on-Write代理
- 保护(Protect or Access)代理
- Cache代理
- 防火墙(Firewall)代理
- 同步化(Synchronization)代理
- 智能引用(Smart Reference)代理
注意事项:
- 和适配器模式区别:适配器模式主要改变所考虑对象的接口,而代理模式不能改变所代理类的接口
- 与装饰器的区别:装饰器模式为了增强功能,而代理模式是为了加以控制
六、优缺点
优点
- 职责清晰
- 扩展性高
- 智能化
缺点
- 由于在客户端和真实主题中间加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢
- 实现代理模式需要额外的工作,有些代理模式的实现非常复杂