ReactiveCocoa源码拆分解析(五)

时间:2022-10-18 18:20:01

(整个关于ReactiveCocoa的代码工程可以在https://github.com/qianhongqiang/QHQReactive下载)

好多天没写东西了,今天继续。主要讲解RAC如何于UI空间实现响应流的。

随手找个按钮响应的RAC实现作为示例,然后我们去做一个简单的实现

[[_HiddenBtn rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {

_HiddenBtn.hidden = YES;

}];

我把一些别的逻辑全部删除,大家也不要关注任何内存问题。这个示例的功能很简单,就是一个按钮被点击抬起这个事件触发后,按钮隐藏。嗯,实现的非常优雅,逻辑也整合到一块去了,代码清晰明了。

我们实现的思路是这样的,创建一个信号,将这个信号与按钮关联,每次触发点击的时候,将值传递给这个信号,沿着信号管道去传递给订阅者。

-(QHQSignal *)qhq_signalForControlEvents:(UIControlEvents)event;我们添加一个UIControl分类,添加类似的方法。

然后考虑一下正常情况下按钮的响应通常是如何添加的,没错,很常见的

[按钮 addTarget:目标对象 action:目标对象方法 forControlEvents:响应事件];

来看看创建信号的方法

+(QHQSignal *)createSignal:(void(^)(id subscriber))didSubscriber

我们之前一直是让这个匿名内建的subscriber去发送消息,比如sendNext等事件,所以顺着这个思路,只需要把addTarget的目标添加为这个subscriber,调用subscriber的sendNext方法就可以了。然后我们来实现一下

-(QHQSignal *)qhq_signalForControlEvents:(UIControlEvents)event {

return [QHQSignal createSignal:^(id<QHQSubscrib> subscriber) {

[self addTarget:subscriber action:@selector(sendNext:) forControlEvents:event];

}];

}

是的,就这一句话就行了。然后来测试以下,我们创建个按钮,然后点击后输出些什么

[[demoButton qhq_signalForControlEvents:UIControlEventTouchDown] subscribeNext:^(id x) {

NSLog(@"%@---被点击了",x);

}];}

结果我点了半天,发现屏幕没有任何输出,怎么回事呢?然后我就开始调试,沿着整个栈信息找,没有什么问题,该创建的都创建了,可以是subscriber的sendNext方法不调用。我重写了一下subscriber的dealloc方法,插了一个断点,预料的一样,进断点了,也就是subscriber被干掉了。

我马上去翻了下API文档,展示一下

// passing in nil as the target goes up the responder chain. The action may optionally include the sender and the event in that order

// the action cannot be NULL. Note that the target is not retained.

- (void)addTarget:(nullable id)target action:(SEL)action forControlEvents:(UIControlEvents)controlEvents;

赫然写着Note that the target is not retained.后来想想,这也是必然的,不然按照常规写法,那不都跟VC循环引用了,自己小白了。

实际上,在RAC中对生命周期的管理做的还是很到位,这里暂时不展开,涉及到的东西比较多。以学习目的为主,我暂时将这个subscriber进行一次retain。

-(QHQSignal *)qhq_signalForControlEvents:(UIControlEvents)event {

return [QHQSignal createSignal:^(id<QHQSubscrib> subscriber) {

[self addTarget:subscriber action:@selector(sendNext:) forControlEvents:event];

objc_setAssociatedObject(self, _cmd, subscriber, OBJC_ASSOCIATION_RETAIN_NONATOMIC);

}];

}

这样代码可以正常运行了。

2016-01-07 13:50:43.142 PageText[10115:8236159] <UIButton: 0x7ff30a771730; frame = (0 200; 320 40); opaque = NO; layer = <CALayer: 0x7ff30a70c930>>---被点击了

在这处理中,自己也是长进不少。剩余的控件大家可以顺着这个思路自己摸索。