iOS开发-消息传递方式-代理(delegate)及协议(protocol)

已经说完了4种消息传递的方式:target-actionKVONotificationCenterblock ,这次我们再说一个一对一的消息传递方式,也是这次消息传递系列的最后一种方式:代理

代理是什么

代理是一种通用的设计模式,在iOS中对代理设计模式有很好的支持,有特定的语法来支持代理模式。

通常代理有三部分组成:

  • 1.协议:用来规定应该做什么,必须做什么。
  • 2.代理方:根据制定的协议,完成委托方需要实现的功能。
  • 3.委托方:根据制定的协议,提出代理方需要实现的功能。

用一张图来表明大家的关系就是:

logo

通过上边的图应该不难理解到,协议就是用来约束代理方和委托方的行为,一般我们都会在协议中写一些方法,委托方来通过协议中的方法传入参数给代理方,代理方可以通过协议完成一些任务,并将结果返回给委托方。

这里举一个例子:比如我需要买一个只有美国才有卖的东西,而我人在中国,也没办法买到美国的东西,这时候我就需要去网上找美国代购小姐姐来帮我购买这个东西,而交易的方式可能就是通过马云爸爸的淘宝店,那么这个时候淘宝中制定的规则就是我们的协议,而我就是委托方,美国代购小姐姐就是代理方,我给代购小姐姐的钱就是我传入的参数,最后他寄给我的东西就是返回的结果。

同理,如果这时候我需要买一个应该才有东西时,我和美国代购小姐姐都没办法做到,而英国代购小哥哥却可以做得到,这时候我就可以委托他来帮我代购我要在英国买的东西,所以一个委托方可以拥有多个代理服务。

代理的使用

接下来就通过一个小例子来简单介绍一下代理的使用方法。

这个例子中我们就简单的使用之前举例的代购来看看代理的使用。我们先把每个委托者也就是代购的同学看做是一个view。

创建协议

首先我们要先创建一个协议,在里边我们可以写好需要做的事情,比如代购的我们就会写一个给钱买东西的方法。

1
2
3
4
5
@protocol MPAmericanBuyDelegate <NSObject>
- (void)buySomethingFromAmericanWithPrice:(CGFloat)aPrice;
@end

这时候还有两个关键字

  • @optional
  • @required

他们的主要作用是用来约束代理是否强制需要遵守协议。默认情况下为@required

但是即使@required的情况下没有遵守协议,编译器也只是会抛出一个警告,并不影响编译。

触发协议条件

接下来,我们需要在写好一个触发回调的条件,比如说代理方(代购)达到某个条件了之后,调用协议中的方法,告诉委托方委托的事情已经做好,并将数据发送给委托方。

这里需要注意一点,就是在调用方法时,需要先判断一下代理的实例是否存在,并且指向的委托方是否能够响应事件,如果委托方没办法响应事件,而代理方发送了结果时,会因为找不到方法的原因而引起崩溃。

1
2
3
4
5
6
//声明
@interface MPAmericanBuyView : UIView
@property (nonatomic ,weak) id<MPAmericanBuyDelegate> delegate;
@end
1
2
3
4
5
6
7
8
9
10
//实现
- (void)finishedBuyBtnClicked
{
//记得先做判断
if(_delegate && [_delegate respondsToSelector:@selector(buySomethingFromAmericanWithPrice:)])
{
[_delegate buySomethingFromAmericanWithPrice:[self.priceField.text floatValue]];
}
}

实现协议方法

现在我们已经处理好了协议和代理方,接下来我们看一下委托方的部分要怎么来实现,首先要遵循协议并建立一个委托方和协议的联系,然后实现回调时委托方要做的事情。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
@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];
}
1
2
3
4
5
6
- (void)buySomethingFromAmericanWithPrice:(CGFloat)aPrice
{
NSLog(@"从美国买好了");
NSLog(@"花了%.2lf元",aPrice);
}

这样一个简单的代理的使用就做好了,在输入了价格,然后点击购买之后,系统就会根据代理找到控制器中的对应方法并调用。

代理的实现原理

代理的本质其实就是代理对象内存的传递和操作,当我们声明了一个代理对象后,实际上只是用了一个id类型(任意类型)的指针将代理对象进行了一个弱引用。

而我们在发送消息给这个代理对象时,实际上就是通过将消息传递给id指针,而这个id指针就是指向代理对象的对象。

就以上边的🌰来说

我们在控制器中将delegate设为了self

1
2
3
MPAmericanBuyView *americanBuy = [[MPAmericanBuyView alloc] initWithFrame:CGRectMake(0, 0, WIDTH, HEIGHT/2)];
americanBuy.backgroundColor = [UIColor brownColor];
americanBuy.delegate = self;

也就是说在MPAmericanBuyView这个类中的delegate指向的就是MPDelegateViewController,所以方法

1
- (void)buySomethingFromAmericanWithPrice:(CGFloat)aPrice;

也被加入到了MPDelegateViewController的方法列表中,这时候调用方法

1
2
3
4
5
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,把之前的几种消息传递方式都放在了里边。地址在这里

最后

几种常见的消息传递方式系列应该就到此告一段落了,所有的文章内容都仅限个人学习参考使用,如果有什么问题还请各位大佬批评指正。