ReactiveCocoa源码拆分解析(五)
(整个关于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>>---被点击了
在这处理中,自己也是长进不少。剩余的控件大家可以顺着这个思路自己摸索。
ReactiveCocoa源码拆分解析(五)的更多相关文章
- ReactiveCocoa源码拆分解析(一)
(整个关于ReactiveCocoa的工程可以在https://github.com/qianhongqiang/QHQReactive下载) ReactiveCocoa的介绍我就不说了,可以自行百度 ...
- ReactiveCocoa源码拆分解析(二)
(整个关于ReactiveCocoa的代码工程可以在https://github.com/qianhongqiang/QHQReactive下载) 上面抽丝剥茧的把最主要的信号机制给分离开了.但在RA ...
- ReactiveCocoa源码拆分解析(四)
(整个关于ReactiveCocoa的代码工程可以在https://github.com/qianhongqiang/QHQReactive下载) 上一章节简要的说明了如何实现的热信号.但是像那么写, ...
- ReactiveCocoa源码拆分解析(七)
(整个关于ReactiveCocoa的代码工程可以在https://github.com/qianhongqiang/QHQReactive下载) 在这篇博客中,我将把ReactiveCocoa中的擦 ...
- ReactiveCocoa源码拆分解析(六)
(整个关于ReactiveCocoa的代码工程可以在https://github.com/qianhongqiang/QHQReactive下载) RAC为了实现优雅的信号绑定,可谓使尽浑身解数,不仅 ...
- ReactiveCocoa源码拆分解析(三)
(整个关于ReactiveCocoa的代码工程可以在https://github.com/qianhongqiang/QHQReactive下载) 这一章节主要讨论信号的“冷”与“热” 在RAC的世界 ...
- 并发编程(十五)——定时器 ScheduledThreadPoolExecutor 实现原理与源码深度解析
在上一篇线程池的文章<并发编程(十一)—— Java 线程池 实现原理与源码深度解析(一)>中从ThreadPoolExecutor源码分析了其运行机制.限于篇幅,留下了Scheduled ...
- Spring框架之spring-web web源码完全解析
Spring框架之spring-web web源码完全解析 spring-web是Spring webMVC的基础,由http.remoting.web三部分组成,核心为web模块.http模块封装了 ...
- 【原】AFNetworking源码阅读(五)
[原]AFNetworking源码阅读(五) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 上一篇中提及到了Multipart Request的构建方法- [AFHTTP ...
随机推荐
- CF731C. Socks[DFS 贪心]
C. Socks time limit per test 2 seconds memory limit per test 256 megabytes input standard input outp ...
- USACO2.4 The Tamworth Two[模拟]
题目描述 两只牛逃跑到了森林里.农夫John开始用他的专家技术追捕这两头牛.你的任务是模拟他们的行为(牛和John). 追击在10x10的平面网格内进行.一个格子可以是: 一个障碍物, 两头牛(它们总 ...
- Hibernate组件映射
Hibernate联合主键映射以及组件映射 在Hibernate中联合主键的形成有两种可能:一种是由多对多映射形成的,多对多映射会形成第三张表,一般来说第三张表的主键是由其他两张表的主键构成的(比如学 ...
- 全国高校网安联赛Web专场~WriteUp
1.Sign 题目:Good Luck!flag{X-nuca@GoodLuck!} Flag直接写在题目上了,flag{X-nuca@GoodLuck!} 2.BaseCoding 提示:这是编码不 ...
- 关于javascript中apply()和call()方法的区别
如果没接触过动态语言,以编译型语言的思维方式去理解javaScript将会有种神奇而怪异的感觉,因为意识上往往不可能的事偏偏就发生了,甚至觉得不可理喻.如果在学JavaScript这自由而变幻无穷的语 ...
- 再谈Newtonsoft.Json高级用法
上一篇Newtonsoft.Json高级用法发布以后收到挺多回复的,本篇将分享几点挺有用的知识点和最近项目中用到的一个新点进行说明,做为对上篇文章的补充. 阅读目录 动态改变属性序列化名称 枚举值序列 ...
- RequireJS中的require如何返回模块
requirejs中定义AMD模块规则如下: define(function(){ var ProductManager={ Create:function(){ console.log(" ...
- JavaScript的理解记录(5)
---接上篇: 三.DOM解析: 1.Document Object Model(DOM):是表示和操作HTML和XML文档内容的基础API;其中几个重要的类有:Document和Element,Te ...
- 微信菜单php 数组格式
//备用 $menu['button']['0']['name'] = '积分'; $menu['button']['0']['sub_button']['0']['type'] = 'view'; ...
- JavaScript错误之:Uncaught ReferenceError: $ is not defined
在js开发中,很多人遇到类似问题,都找不到解决方法.Uncaught ReferenceError: $ is not defined,在这里给大家提供几个解决方法. 方法/步骤11.出现这个错误,最 ...