已经说完了4种消息传递的方式:target-action 、KVO 、NotificationCenter 、block ,这次我们再说一个一对一的消息传递方式,也是这次消息传递系列的最后一种方式:代理。
代理是什么
代理是一种通用的设计模式,在iOS中对代理设计模式有很好的支持,有特定的语法来支持代理模式。
通常代理有三部分组成:
1.协议:用来规定应该做什么,必须做什么。
2.代理方:根据制定的协议,完成委托方需要实现的功能。
3.委托方:根据制定的协议,提出代理方需要实现的功能。
用一张图来表明大家的关系就是:
通过上边的图应该不难理解到,协议就是用来约束代理方和委托方的行为,一般我们都会在协议中写一些方法,委托方来通过协议中的方法传入参数给代理方,代理方可以通过协议完成一些任务,并将结果返回给委托方。
这里举一个例子:比如我需要买一个只有美国才有卖的东西,而我人在中国,也没办法买到美国的东西,这时候我就需要去网上找美国代购小姐姐来帮我购买这个东西,而交易的方式可能就是通过马云爸爸的淘宝店,那么这个时候淘宝中制定的规则就是我们的协议,而我就是委托方,美国代购小姐姐就是代理方,我给代购小姐姐的钱就是我传入的参数,最后他寄给我的东西就是返回的结果。
同理,如果这时候我需要买一个应该才有东西时,我和美国代购小姐姐都没办法做到,而英国代购小哥哥却可以做得到,这时候我就可以委托他来帮我代购我要在英国买的东西,所以一个委托方可以拥有多个代理服务。
代理的使用
接下来就通过一个小例子来简单介绍一下代理的使用方法。
这个例子中我们就简单的使用之前举例的代购来看看代理的使用。我们先把每个委托者也就是代购的同学看做是一个view。
创建协议
首先我们要先创建一个协议,在里边我们可以写好需要做的事情,比如代购的我们就会写一个给钱买东西的方法。
@protocol MPAmericanBuyDelegate <NSObject>
- (void)buySomethingFromAmericanWithPrice:(CGFloat)aPrice;
@end
这时候还有两个关键字
@optional
@required
他们的主要作用是用来约束代理是否强制需要遵守协议。默认情况下为@required
。
但是即使@required
的情况下没有遵守协议,编译器也只是会抛出一个警告,并不影响编译。
触发协议条件
接下来,我们需要在写好一个触发回调的条件,比如说代理方(代购)达到某个条件了之后,调用协议中的方法,告诉委托方委托的事情已经做好,并将数据发送给委托方。
这里需要注意一点,就是在调用方法时,需要先判断一下代理的实例是否存在,并且指向的委托方是否能够响应事件,如果委托方没办法响应事件,而代理方发送了结果时,会因为找不到方法的原因而引起崩溃。
//声明
@interface MPAmericanBuyView : UIView
@property (nonatomic ,weak) id<MPAmericanBuyDelegate> delegate;
@end
//实现
- (void)finishedBuyBtnClicked
{
//记得先做判断
if(_delegate && [_delegate respondsToSelector:@selector(buySomethingFromAmericanWithPrice:)])
{
[_delegate buySomethingFromAmericanWithPrice:[self.priceField.text floatValue]];
}
}
实现协议方法
现在我们已经处理好了协议和代理方,接下来我们看一下委托方的部分要怎么来实现,首先要遵循协议并建立一个委托方和协议的联系,然后实现回调时委托方要做的事情。
@interface MPDelegateViewController ()<MPAmericanBuyDelegate>
@end
@implementation MPDelegateViewController
- (void)viewDidLoad
{
[super viewDidLoad];
self.view.backgroundColor = [UIColor whiteColor];
MPAmericanBuyView *americanBuy = [[MPAmericanBuyView alloc] initWithFrame:CGRectMake(0, 0, WIDTH, HEIGHT/2)];
americanBuy.backgroundColor = [UIColor brownColor];
americanBuy.delegate = self; //设置代理
[self.view addSubview:americanBuy];
}
- (void)buySomethingFromAmericanWithPrice:(CGFloat)aPrice
{
NSLog(@"从美国买好了");
NSLog(@"花了%.2lf元",aPrice);
}
这样一个简单的代理的使用就做好了,在输入了价格,然后点击购买之后,系统就会根据代理找到控制器中的对应方法并调用。
代理的实现原理
代理的本质其实就是代理对象内存的传递和操作,当我们声明了一个代理对象后,实际上只是用了一个****id****类型(任意类型)的指针将代理对象进行了一个弱引用。
而我们在发送消息给这个代理对象时,实际上就是通过将消息传递给id指针,而这个id指针就是指向代理对象的对象。
就以上边的🌰来说
我们在控制器中将delegate设为了self
MPAmericanBuyView *americanBuy = [[MPAmericanBuyView alloc] initWithFrame:CGRectMake(0, 0, WIDTH, HEIGHT/2)];
americanBuy.backgroundColor = [UIColor brownColor];
americanBuy.delegate = self;
也就是说在MPAmericanBuyView
这个类中的delegate
指向的就是MPDelegateViewController
,所以方法
- (void)buySomethingFromAmericanWithPrice:(CGFloat)aPrice;
也被加入到了MPDelegateViewController
的方法列表中,这时候调用方法
if(_delegate && [_delegate respondsToSelector:@selector(buySomethingFromAmericanWithPrice:)])
{
[_delegate buySomethingFromAmericanWithPrice:[self.priceField.text floatValue]];
}
时就会去MPDelegateViewController
的方法列表中来找对应方法名的方法,如果没有,就会发生找不到方法的崩溃。
循环引用
既然是这样,那就有可能会出现一个循环引用的问题。如果代理方强引用了委托方的对象,而委托方又强引用了delegate属性。那这时候两者就会出现循环引用,都无法正常释放。
这时候就需要我们把delegate属性设置为弱引用,这样在代理的生命周期内,他还是可以正常工作的,在结束了自己的工作之后,因为没有出现强引用,也不会产生循环引用的问题。
但是一定要用weak么?用assign不可以么?
首先说明,他们两个用来修饰变量都不会改变引用对象的引用计数,但是在一个对象被释放后,weak会自动将指针指向nil,在iOS中,向nil发送消息是不会产生崩溃的,但是assign则会产生一个野指针,这时候如果想他发送消息,就会出现找不到方法的问题而崩溃。
所以在使用时最好还是用weak来修饰delegate。
Demo
好了,代理的基本使用和简单的原理就说到这里了,最后上一个demo,把之前的几种消息传递方式都放在了里边。地址在这里
最后
几种常见的消息传递方式系列应该就到此告一段落了,所有的文章内容都仅限个人学习参考使用,如果有什么问题还请各位大佬批评指正。