理解代理模式

1.代理模式的主旨

要想实现代理模式,就要有三个组成部分,一个是协议,一个是代理者,一个是委托者。

我们需要定义一套接口(协议),如果一个对象a想要接受另一个对象b的委托帮助他做事,对象a就要遵循这套接口(协议),a对象就成为了b对象的代理。b对象就可以给a对象传递信息,也可以在发生事件时通知a对象。

2.使用代理模式的方法

1.首先要明确谁是代理者,谁是委托者。在委托者类的头文件里我们生成协议,协议里只有方法的声明,没有方法的实现,协议里的方法可以用关键字@optional和@required来区分是否必须实现。

@protocol secondViewControllerDelegate

-(void)sendString:(NSString*)value;

@end

2.然后,声明一个属性来存放代理对象

@property(nonatomic,weak)id delegate;

⚠️:这个属性一定要声明为weak,为什么呢,因为代理对象a会持有本对象b,如果b再持有代理对象的话,就会造成循环引用,所以用weak修饰,还可以做到在相关对象销毁时自动清空。

3.在委托对象里调用代理方法,在调用代理方法之前应该先检查一下代理对象有没有实现这个方法。

if([_delegate respondsToSelector:@selector(sendString:)]){

    [_delegate sendString:@"i am secondVC"];

    }

 如果实现了就调用,如果没实现就什么也不做,避免因为代理对象没实现方法而出现问题。如果调用了一个代理对象没实现的方法,程序不会crash,传回来的值是null。

4.代理对象遵从协议,设置代理,并实现协议里声明的方法。

@interface ViewController ()

@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;

}

你可能感兴趣的:(理解代理模式)