利用runtime实现多重代理

在Objective-C中,经常使用delegate来在对象之间通信,但是

delegate一般是对象间一对一的通信,有时候我们希望delegate方法由多个不同的对象来处理,比如UITableView继承于

UIScrollView,我们希望他的delegate中UIScrollViewDelegate的方法由一个独立的类来处理,以便实现一些效果,比

如像下图这样的头部图片滚动拉伸效果,只需要实现UIScrolViewDelegate的scrollViewDidScroll方法,这样做的好处是

可以降低代码耦合度,将实现不同功能的方法封装在独立的delegate中,便于复用和维护管理。

一、OC的消息机制

么,怎样实现delegate方法的动态转发呢?这需要用到Objective-C的消息机制,我们都知道,在OC中,调用一个对象的方法,实际上是给对

象发了一条消息,在编译Objective-C函数调用的语法时,会被翻译成一个C的函数调用:objc_msgSend(),例如:


//会被翻译成:

objc_msgSend(array, @selector(insertObject:atIndex), foo, 2);

那么,objc_msgSend又做了哪些事呢?,以[object foo]为例:

通过object的isa指针找到它的class

在class的method_list中找到foo

如果class中没找到foo,则继续往他的superclass中查找

一旦找到foo这个函数,就去执行对应的方法实现(IMP)

如果一直没有找到foo,OC的runtime将继续下面的步骤:

二、动态方法决议与消息转发

在Objective-C中,如果向一个对象发送一条该对象无法处理的消息(对应selector不存在),会导致程序crash, 但是,在crash之前,oc的运行时系统会先经过以下两个步骤:

Dynamic Method Resolution

Message Forwarding

1、Dynamic Method Resolution(动态方法决议)

首先,如果调用的方法是实例方法,OC的运行时会调用-

(BOOL)resolveInstanceMethod:(SEL)sel,如果是类方法,则会调用+

(BOOL)resolveClassMethod:(SEL)sel

让我们可以在程序运行时动态的为一个selector提供实现,如果我们添加了函数的实现,并返回YES,运行时系统会重启一次消息的发送过程,调用动态

添加的方法。例如,下面的例子:

1

2

3

4

5

6

7

8

9

10+ (BOOL)resolveInstanceMethod:(SEL)sel{

if(sel == @selector(foo)) {

class_addMethod([self class], sel, (IMP)dynamicMethodIMP,"V@:");

returnYES;

}

return[superresolveInstanceMethod:sel];

}

void dynamicMethodIMP(id self, SEL _cmd){

NSLog(@"%s", __PRETTY_FUNCTION__);

}

class_addMethod

方法动态的添加新的方法与对应的实现,如果调用了[Foo foo],将会转到动态添加的dynamicMethodIMP

方法中。Objective-C的方法本质上是一个至少包含两个参数(id self, SEL

_cmd)的C函数,这样,当重启消息发送时,就能在类中找到@selector(foo)了。而如果方法返回NO时,将会进入下一步:消息转发

(Message Forwarding)

2、Message Forwarding(消息转发)

消息转发分为两步:

首先运行时系统会调用- (id)forwardingTargetForSelector:(SEL)aSelector方法,如果这个方法中返回的不是nil或者self,运行时系统将把消息发送给返回的那个对象

如果

-(id)forwardingTargetForSelector:(SEL)aSelector返回的是nil或者self,运行时系统首先会调用-

(NSMethodSignature

*)methodSignatureForSelector:(SEL)aSelector方法来获得方法签名,方法签名记录了方法的参数和返回值的信

息,如果-methodSignatureForSelector 返回的是nil, 运行时系统会抛出unrecognized selector

exception,程序到这里就结束了。

整个流程可以用下面这张图表示

三、实现多重代理

结合上面的流程分析,我么可以发现,要实现多重代理的分发,我们需要让Runtime系统运行到forwardInvocation这一步,并在该方法中将delegate方法分发到其他各个对象中去:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{

NSMethodSignature *signature = [supermethodSignatureForSelector:aSelector];

if(!signature) {

for(id targetinself.allDelegates) {

if((signature = [target methodSignatureForSelector:aSelector])) {

break;

}

}

}

returnsignature;

}

- (void)forwardInvocation:(NSInvocation *)anInvocation{

for(id targetinself.allDelegates) {

if([target respondsToSelector:anInvocation.selector]) {

[anInvocation invokeWithTarget:target];

}

}

}

由于我们调用delegate的方法时,一般会先调用[delegate responseToSelector]方法,所以,我们还需要实现这个方法:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15- (BOOL)respondsToSelector:(SEL)aSelector{

if([superrespondsToSelector:aSelector]) {

returnYES;

}

for(id targetinself.allDelegates) {

if([target respondsToSelector:aSelector]) {

returnYES;

}

}

returnNO;

}

@end

然后我们来测试一下,新建一个ScrollDelegate类,实现UIScrollViewDelegate方法:

1

2

3

4

5

6

7

8

9#import "ScrollDelegate.h"

@implementation ScrollDelegate

- (void)scrollViewDidScroll:(UIScrollView *)scrollView{

NSLog(@"%s", __PRETTY_FUNCTION__);

}

@end

然后再新建一个ViewController,也实现UIScrollViewDelegate方法,添加一个UIScrollView在controller的view中,然后设置scrollView的delegate为multipleProxy:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30#import "ViewController.h"

#import "MultipleDelegate.h"

#import "ScrollDelegate.h"

@interface ViewController ()@property (weak, nonatomic) IBOutlet UIScrollView *scrollView;

@end

@implementation ViewController{

MultipleDelegate *_multipleDelegate;

}

- (void)viewDidLoad {

[superviewDidLoad];

self.scrollView.contentSize = CGSizeMake(self.view.bounds.size.width, self.view.bounds.size.height * 2);

_multipleDelegate = [MultipleDelegatenew];

//添加要处理delegate方法的对象

NSArray *array = @[self, [ScrollDelegatenew]];

_multipleDelegate.allDelegates = array;

self.scrollView.delegate = (id)_multipleDelegate;

}

- (void)scrollViewDidScroll:(UIScrollView *)scrollView{

NSLog(@"%s", __PRETTY_FUNCTION__);

}

@end

运行,滑动scrollView,看看控制台打印的信息:

1

2

3

42015-11-01 11:07:49.199 MultipleDelegateDemo[4732:498520] -[ViewController scrollViewDidScroll:]

2015-11-01 11:07:49.200 MultipleDelegateDemo[4732:498520] -[ScrollDelegate scrollViewDidScroll:]

2015-11-01 11:07:49.227 MultipleDelegateDemo[4732:498520] -[ViewController scrollViewDidScroll:]

2015-11-01 11:07:49.227 MultipleDelegateDemo[4732:498520] -[ScrollDelegate scrollViewDidScroll:]

好,deegate方法已经被正确地转发给了两个对象了,看起来好像没什么不对,可是,细心的你一定会发现,这里存在retain

cycle:controller -> _multipleDelegate -> controller,那么怎样解决这个问题呢?

四、NSPointerArray防止循环引用

为NSArray会对对象进行retain操作,导致循环引用的产生,所以我们可以用NSPointerArray来解决这个问题,但是需要注意对于其他

的delegate对象也需要在controller中对其强引用, 最终MultipleDelegateProxy的实现:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51#import "KIZMultipleDelegateProxy.h"

@interface KIZMultipleDelegateProxy ()

@property (nonatomic, strong) NSPointerArray *weakRefTargets;

@end

@implementation KIZMultipleDelegateProxy

- (void)setDelegateTargets:(NSArray *)delegateTargets{

self.weakRefTargets = [NSPointerArray weakObjectsPointerArray];

for(id delegateindelegateTargets) {

[self.weakRefTargets addPointer:(__bridge void *)delegate];

}

}

- (BOOL)respondsToSelector:(SEL)aSelector{

if([superrespondsToSelector:aSelector]) {

returnYES;

}

for(id targetinself.weakRefTargets) {

if([target respondsToSelector:aSelector]) {

returnYES;

}

}

returnNO;

}

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{

NSMethodSignature *sig = [supermethodSignatureForSelector:aSelector];

if(!sig) {

for(id targetinself.weakRefTargets) {

if((sig = [target methodSignatureForSelector:aSelector])) {

break;

}

}

}

returnsig;

}

//转发方法调用给所有delegate

- (void)forwardInvocation:(NSInvocation *)anInvocation{

for(id targetinself.weakRefTargets) {

if([target respondsToSelector:anInvocation.selector]) {

[anInvocation invokeWithTarget:target];

}

}

}

@end

五、小记

利用这个多重代理动态转发,我封装了一些独立的delegate实现的小功能,比如本文开头提到的TableView头部图片拉伸效果,放在github上:https://github.com/zziking/KIZBehavior

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,717评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,501评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,311评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,417评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,500评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,538评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,557评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,310评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,759评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,065评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,233评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,909评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,548评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,172评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,420评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,103评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,098评论 2 352

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,698评论 0 9
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,188评论 0 7
  • 转载:http://yulingtianxia.com/blog/2014/11/05/objective-c-r...
    F麦子阅读 730评论 0 2
  • 本文转载自:http://yulingtianxia.com/blog/2014/11/05/objective-...
    ant_flex阅读 755评论 0 1
  • runtime 和 runloop 作为一个程序员进阶是必须的,也是非常重要的, 在面试过程中是经常会被问到的, ...
    made_China阅读 1,210评论 0 7