iOS 设计模式之十三(代理模式)
一、概念
1、代理模式的动机
随着贸易的发展,网络代购成为一道亮丽的风景线。顾客本地买不到需要的产品,便找代购去外国把产品买回来,也就是顾客只需要和代购交流便可以买产品了。
在软件开发中,也有一种设计模式可以提供与代购类似的功能。由于某些原因,客户端不想或不能直接访问一个对象,此时可以通过一个称之为“代理”的第三者来实现间接访问,该方案对应的设计模式被称为代理模式。
2、代理模式的定义
代理模式(Proxy Pattern):给某一个对象提供一个代理或占位符,并由代理对象来控制对原对象的访问。
3、代理模式的3个角色
1)Subject(抽象主题角色):它声明了真实主题和代理主题的共同接口,这样一来在任何使用真实主题的地方都可以使用代理主题,客户端通常需要针对抽象主题角色进行编程。
2)Proxy(代理主题角色):它包含了对真实主题的引用,从而可以在任何时候操作真实主题对象;在代理主题角色中提供一个与真实主题角色相同的接口,以便在任何时候都可以替代真实主题。代理主题角色还可以控制对真实主题的使用,负责在需要的时候创建和删除真实主题对象,并对真实主题对象的使用加以约束。
3)RealSubject(真实主题角色):它定义了代理角色所代表的真实对象,在真实主题角色中实现了真实的业务操作,客户端可以通过代理主题角色间接调用真实主题角色中定义的操作。
4、结构图
代理模式5、常用的代理模式分类
1)远程代理(Remote Proxy):为一个位于不同的地址空间的对象提供一个本地的代理对象,这个不同的地址空间可以是在同一台主机中,也可是在另一台主机中,远程代理又称为大使(Ambassador)。
2)虚拟代理(Virtual Proxy):如果需要创建一个资源消耗较大的对象,先创建一个消耗相对较小的对象来表示,真实对象只在需要时才会被真正创建。
3)保护代理(Protect Proxy):控制对一个对象的访问,可以给不同的用户提供不同级别的使用权限。
4)缓冲代理(Cache Proxy):为某一个目标操作的结果提供临时的存储空间,以便多个客户端可以共享这些结果。
5)智能引用代理(Smart Reference Proxy):当一个对象被引用时,提供一些额外的操作,例如将对象被调用的次数记录下来等。
二、示例
1)先创建一个BuySomething协议,协议有两个买东西的方法,表示抽象主题角色;
2)然后创建一个Person类,遵守BuySomething协议,表示真实主题角色;
3)最后创建一个Proxy类,里面有一个Person对象,也遵守BuySomething协议,表示代理主题角色;
BuySomething协议:
@protocol BuySomething <NSObject>
- (void)buyLV; //购买LV
- (void)buyChanel; //购买香奈儿
@end
Person类:
@interface Person : NSObject<BuySomething>
@property (nonatomic, copy) NSString *name;
- (instancetype)initWithName:(NSString *)name;
@end
@implementation Person
- (instancetype)initWithName:(NSString *)name {
self = [super init];
if (self) {
_name = name;
}
return self;
}
- (void)buyLV {
NSLog(@"%@购买了LV包包", self.name);
}
- (void)buyChanel {
NSLog(@"%@购买了香奈儿香水", self.name);
}
@end
Proxy类:
// 代购商
// .h文件
@interface Proxy : NSObject<BuySomething>
- (instancetype)initWithPerson:(Person *)person;
@end
// .m文件
@interface Proxy ()
@property (nonatomic, strong) Person *person;
@end
@implementation Proxy
- (instancetype)initWithPerson:(Person *)person {
self = [super init];
if (self) {
_person = person;
}
return self;
}
- (void)buyLV {
[self.person buyLV];
NSLog(@"%@买包包送手环", self.person.name); //可以增加额外的操作
}
- (void)buyChanel {
[self.person buyChanel];
}
@end
运行代码:
- (void)viewDidLoad {
[super viewDidLoad];
Person *xiaoHong = [[Person alloc] initWithName:@"小红"];
Proxy *proxy1 = [[Proxy alloc] initWithPerson:xiaoHong];
[proxy1 buyChanel];
[proxy1 buyLV];
NSLog(@"--------------");
Person *xiaoFang = [[Person alloc] initWithName:@"小芳"];
Proxy *proxy2 = [[Proxy alloc] initWithPerson:xiaoFang];
[proxy2 buyChanel];
}
打印结果:
小红购买了香奈儿香水
小红购买了LV包包
小红买包包送手环
--------------
小芳购买了香奈儿香水
三、总结
代理模式为对象的间接访问提供了一个解决方案,可以对对象的访问进行控制。
1、优点
共同优点:
1、能够协调调用者和被调用者,在一定程度上降低了系统的耦合度。
2、客户端可以针对抽象主题角色进行编程,增加和更换代理类无须修改源代码,符合开闭原则,系统具有较好的灵活性和可扩展性。
不同类型的代理模式的独特优点:
1、远程代理为位于两个不同地址空间对象的访问提供了一种实现机制,可以将一些消耗资源较多的对象和操作移至性能更好的计算机上,提高系统的整体运行效率。
2、虚拟代理通过一个消耗资源较少的对象来代表一个消耗资源较多的对象,可以在一定程度上节省系统的运行开销。
3、缓冲代理为某一个操作的结果提供临时的缓存存储空间,以便在后续使用中能够共享这些结果,优化系统性能,缩短执行时间。
4、保护代理可以控制对一个对象的访问权限,为不同用户提供不同级别的使用权限。
2、缺点
1、由于在客户端和真实主题之间增加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢,例如保护代理。
2、实现代理模式需要额外的工作,而且有些代理模式的实现过程较为复杂,例如远程代理。
3、适用场景
1、当客户端对象需要访问远程主机中的对象时可以使用远程代理。
2、当需要用一个消耗资源较少的对象来代表一个消耗资源较多的对象,从而降低系统开销、缩短运行时间时可以使用虚拟代理,例如一个对象需要很长时间才能完成加载时。
3、当需要为某一个被频繁访问的操作结果提供一个临时存储空间,以供多个客户端共享访问这些结果时可以使用缓冲代理。通过使用缓冲代理,系统无须在客户端每一次访问时都重新执行操作,只需直接从临时缓冲区获取操作结果即可。
4、当需要控制对一个对象的访问,为不同用户提供不同级别的访问权限时可以使用保护代理。
5、当需要为一个对象的访问(引用)提供一些额外的操作时可以使用智能引用代理。
4、iOS应用举例
大家都很熟悉UITableView的delegate和dataSource,都是通过代理对象来控制对UITableView的访问。
Demo地址:iOS-Design-Patterns