1.代理模式的主旨
要想实现代理模式,就要有三个组成部分,一个是协议,一个是代理者,一个是委托者。
我们需要定义一套接口(协议),如果一个对象a想要接受另一个对象b的委托帮助他做事,对象a就要遵循这套接口(协议),a对象就成为了b对象的代理。b对象就可以给a对象传递信息,也可以在发生事件时通知a对象。
2.使用代理模式的方法
1.首先要明确谁是代理者,谁是委托者。在委托者类的头文件里我们生成协议,协议里只有方法的声明,没有方法的实现,协议里的方法可以用关键字@optional和@required来区分是否必须实现。
@protocol secondViewControllerDelegate<NSObject>
-(void)sendString:(NSString*)value;
@end
2.然后,声明一个属性来存放代理对象。
@property(nonatomic,weak)id<secondViewControllerDelegate> delegate;
⚠️:这个属性一定要声明为weak,为什么呢,因为代理对象a会持有本对象b,如果b再持有代理对象的话,就会造成循环引用,所以用weak修饰,还可以做到在相关对象销毁时自动清空。
3.在委托对象里调用代理方法,在调用代理方法之前应该先检查一下代理对象有没有实现这个方法。
if([_delegate respondsToSelector:@selector(sendString:)]){
[_delegate sendString:@"i am secondVC"];
}
如果实现了就调用,如果没实现就什么也不做,避免因为代理对象没实现方法而出现问题。如果调用了一个代理对象没实现的方法,程序不会crash,传回来的值是null。
4.代理对象遵从协议,设置代理,并实现协议里声明的方法。
@interface ViewController ()<secondViewControllerDelegate>
@end
SecondViewController* secondVC = [[SecondViewController alloc] init];
secondVC.delegate = self;
🌹:这句话我终于懂了!delegate其实是secondVC的一个属性,这个属性的类型是secondViewControllerDelegate,相当于把委托对象的代理属性设置为代理对象(self)。
代理对象才是真正做事的人,实现方法。
-(void)sendString:(NSString *)value{
NSLog(@"%@",value);
}
3.关于代理对象是否实现某个协议方法的优化
如果在程序的执行过程中,某一个协议方法需要被多次执行,那么如果每次都通过respondsToSelector方法去检查代理对象有没有实现这个方法是没有必要的,因为这个结果其实在第一次检查的时候就已经确定了,并且之后也不会改变。因此我们可以想办法把这个结果缓存起来,不用每次都去检查。
实现的方法就是在委托对象里声明一个结构体,例如
{
struct{
unsigned int sendString:1;
}_delegateFlags;
}
第一次调用respondsToSelector方法的时候就把值赋值给结构体里的值,
_delegateFlags.sendString = [_delegate responseToSelector:(@selector(sendString))];
这样每次调用相关方法之前就不需要用responseToSelector去检查了:
if(_delegateFlags.sendString){
}
4.代理和NSNotification(通知)的区别
代理一般都是一对一的,通知可以实现多个对象间的通信。
5.如何用代理实现一对多
为一个对象添加多个代理对象。
声明普通的代理属性默认会生成一个set方法,用set方法没办法设置多代理,因为新的代理对象会把原来的代理对象覆盖掉,永远都只有一个。
需要我们手动写一个添加代理对象的方法,把代理对象添加到一个数组中,但是数组会引用代理对象导致代理无法被释放,所以要用一个桥接。
-(void)setDelegate:(id)delegate
{
if ([delegate respondsToSelector:@selector(response:)]) {
[self.weakRefTargets addPointer:(__bridge void *)delegate];
}
}
-(NSPointerArray *)weakRefTargets
{
if (!_weakRefTargets) {
_weakRefTargets = [NSPointerArray weakObjectsPointerArray];
}
return _weakRefTargets;
}
网友评论