协议与代理的异形变换

前言

一个类/对象只要遵守某个协议就可以调用协议方法,从而在某些方面达成共识。如果单纯遵守协议并实现协议方法,在某些场景从外部调用,这篇文章已经没有存在的必要了。当协议与代理配合使用时,可以组成代理模式。在iOS中有这么一句话,“代理是一对一的”,本文围绕这句话展开,并给出不同场景下的多种解决方案及终极解决方案。

引例

场景:假设JKScrollView继承自UIScrollView,JKScrollView对外提供的接口需要用到UIScrollViewDelegate,此时需要将JKScrollView对应实例对象的代理设置为自身。这样一来,当外部重设代理时,会导致内部代理失效。怎样在确保内部代理正常的前提下,外部仍然可以获取代理相应的功能?

写法一:

最简单的写法大概长这样:

NS_ASSUME_NONNULL_BEGIN

@class JKScrollView;
typedef void(^JKScrollViewDidScrollBlock)(JKScrollView *scrollView);

@interface JKScrollView : UIScrollView<UIScrollViewDelegate>
@property (nullable, nonatomic, copy) JKScrollViewDidScrollBlock didScrollBlock;
@end

NS_ASSUME_NONNULL_END
- (instancetype)init {
    if (self = [super init]) {
        [super setDelegate:self];
    }
    return self;
}

- (void)setDelegate:(id<UIScrollViewDelegate>)delegate {
    if (!delegate || self.delegate == delegate) return;
    [super setDelegate:self];
}

- (void)setDidScrollBlock:(JKScrollViewDidScrollBlock)didScrollBlock {
    if (!didScrollBlock) return;
    _didScrollBlock = [didScrollBlock copy];
}

- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
    !_didScrollBlock ? : _didScrollBlock(self);
    //  do something
}

1.在初始化时,将代理设为self
2.重写代理的setter,强制将代理设为self(以防外部改变delegate)
3.在代理方法中执行外部传入的block

这样写确实可以完成相应需求,但是需要手动添加block,如果不想添加block也可以这么写


写法二:
NS_ASSUME_NONNULL_BEGIN

@interface JKTableScrollView : UIScrollView
@property (nullable, nonatomic, weak, readonly) id<UIScrollViewDelegate> fakeDelegate;
@end

NS_ASSUME_NONNULL_END
- (instancetype)init {
    if (self = [super init]) {
        [super setDelegate:self];
    }
    return self;
}

- (void)setDelegate:(id<UIScrollViewDelegate>)delegate {
    if (!delegate || self.delegate == delegate) return;
    _fakeDelegate = delegate;
    [super setDelegate:self];
}

- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
    if (_fakeDelegate && [_fakeDelegate respondsToSelector:@selector(scrollViewDidScroll:)]) {
        [_fakeDelegate scrollViewDidScroll:scrollView];
    }
    // do something
}

与第一种方法略有不同,在delegate的setter中,将外部设置的delegate保存,并在内部代理方法调用时判断外部代理是否实现相同方法,如果实现,则手动调用。
这两种写法思想类似,并且写起来都很麻烦,特别当有多个代理方法要实现,会做很多无用功。

而且会有这么一种情况:
内部并不需要用到代理的某个方法,但是外部需要用到,此时在内部还得写相应的代理方法用来适配外部代理调用。

为了减少这种适配,现在引入第三种写法。
在介绍第三种写法前,请先确保了解消息转发流程,如果不了解,可以看这篇文章传送门

除此之外,还会用到runtime中的这个函数

struct objc_method_description protocol_getMethodDescription(Protocol * _Nonnull proto, SEL _Nonnull aSel, BOOL isRequiredMethod, BOOL isInstanceMethod)

可以看到,objc_method_description结构体的成员变量很简单,只有方法名与参数

objc_method_description


写法三:

说明:这种方法在第二种方法的基础上添加如下代码

- (BOOL)respondsToSelector:(SEL)aSelector {
    BOOL result = [super respondsToSelector:aSelector];
    if (!result && _fakeDelegate) {
        struct objc_method_description omd = protocol_getMethodDescription(@protocol(UIScrollViewDelegate), aSelector, NO, YES);
        if (omd.name) {
            result = [_fakeDelegate respondsToSelector:aSelector];
        }
    }
    return result;
}

- (id)forwardingTargetForSelector:(SEL)aSelector {
    if (_fakeDelegate) {
        struct objc_method_description omd = protocol_getMethodDescription(@protocol(UIScrollViewDelegate), aSelector, NO, YES);
        if (omd.name) {
            return _fakeDelegate;
        }
    }
    return [super forwardingTargetForSelector:aSelector];
}

重写respondsToSelector :方法,满足代码中条件则认为可响应。若某代理方法内部未实现,而外部代理实现(fakeDelegate),由于真实代理为内部代理(self),正常流程fakeDelegate无法响应。重写respondsToSelector :后会进入消息转发流程,在forwardingTargetForSelector:判断是否满足条件,如果满足则转发给_fakeDelegate,从而使_fakeDelegate可以响应内部未实现的代理方法。

这种写法相对前两种方法大大简化了实现外部代理的书写流程,但还是很烦,因为内部代理已实现的方法没法进入消息转发流程,所以只要内部代理实现的方法,外部代理想实现就必须在内部做一次判断。

同时还有这么两种情况:
如果外部要实现的代理有多个怎么办?
如果内部并不需要实现代理,外部需要实现多个代理又怎么办?


接着方法三的思路来,是否能在消息转发流程将要实现的代理方法转发给多个外部代理对象?显然是可以的,因为消息转发的最后一步forwardInvocation:参数为NSInvocation,而NSInvocation可以指定target。如果不熟悉NSInvocation,可以看这篇文章传送门
在开始终极写法前,还需要解决一个问题:targets由外部传入,如果直接用NSArray保存,而保存的对象又被其他对象持有,此时极易导致循环引用。因此,这里用NSPointerArray代替NSArray以防循环引用

终极写法:

直接上代码:

#import "ViewController.h"
#import "JKProtocolHelper.h"

@interface Test: NSObject<UIScrollViewDelegate>
@end

@implementation Test
- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
    NSLog(@"Test");
}
@end


@interface ViewController ()<UITableViewDelegate>
@property (nonatomic, strong) Test *test;
@property (nonatomic, strong) id helper;
@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    UITableView *tableView = [[UITableView alloc] initWithFrame:self.view.bounds style:UITableViewStylePlain];
    [self.view addSubview:tableView];
    
    _test = [Test new];
    _helper = [JKProtocolHelper helperWithProtocol:@protocol(UIScrollViewDelegate) executors:@[self, _test]];
    tableView.delegate = _helper;
}

- (void)dealloc {
    NSLog(@"%s", __func__);
}

- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
    NSLog(@"view controller");
}


@end
JKProtocolHelper.gif

Demo已放到github,自取

Have fun!

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,698评论 0 9
  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,939评论 6 13
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • 自从那次我失恋了,我就把我自己关在屋里,不想出门被美女劈腿,从此变成了宅男,宅在屋里干啥?玩游戏、看直播呗!…… ...
    韩樟树阅读 10,640评论 1 1
  • 平静地生活只会索然无趣,于是,麻烦就来了。 我与女儿住在单位,爱人下班后也来与我们同住。这一天,他...
    秋之美阅读 285评论 0 2