ReactiveCocoa信号使用方法
最近研究RAC时都是基于UI控件来使用,对单独的signal没有使用过,最近在网上看到一篇文章是关于RAC单独signal的使用。在学习整理后将个人觉得能帮助用于UI控件的一些signal使用方法记录如下(也许能从中思考出用于UI控件信号组合的方法):
1.基本的signal创建使用
//创建一个signal,并直接发送next事件对象
RACSignal *signalA = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
NSLog(@"1====");
//代表信号可以使用next事件,那么接收的对象为“你确定”
[subscriber sendNext:@"你确定"];
//代表此信号可以使用completed事件
[subscriber sendCompleted];
//返回一个空对象
return nil;
}];
//信号的next事件同completed事件
[signalA subscribeNext:^(id x) {
NSLog(@"2====%@",x);
}];
[signalA subscribeCompleted:^{
NSLog(@"3====消息提交完成");
}];
[signalA subscribeNext:^(id x) {
NSLog(@"4====我还没有消失");
}];
[signalA subscribeCompleted:^{
NSLog(@"5====信号还在");
}];
接下来是打印结果:让我们从结果分析其运行过程,首先创建信号,并没有走block信号设置(3-8行代码),当其发送next事件时,便会回到block信号设置中查询此信号有没有要求发送next事件,如果没有的话,其next事件便不会被触发。completed事件同next一样。那说明单独创建的signal中你需要设置的是此信号可以执行的事件有哪些!注意,block信号设置中sendcompleted后写的事件不会被执行(读者可以试着调换block信号设置中的代码顺序查看结果)
2.信号队列的使用
信号队列顾名思义就是将一组信号排成队列,挨个调用,下面我们来看代码
//创建3个信号来模拟队列
RACSignal *signalB = [RACSignal createSignal:^RACDisposable *(id subscriber) {
[subscriber sendNext:@"喜欢一个人"];
[subscriber sendCompleted];
return nil;
}];
RACSignal *signalC = [RACSignal createSignal:^RACDisposable *(id subscriber) {
[subscriber sendNext:@"直接去表白"];
[subscriber sendCompleted];
return nil;
}];
RACSignal *signalD = [RACSignal createSignal:^RACDisposable *(id subscriber) {
[subscriber sendNext:@"成功在一起"];
[subscriber sendCompleted];
return nil;
}];
//连接组队列:将几个信号放进一个组里面,按顺序连接每个,每个信号必须执行sendCompleted方法后才能执行下一个信号
RACSignal *signalGroup = [[signalB concat:signalC] concat:signalD];
[signalGroup subscribeNext:^(id x) {
NSLog(@"%@",x);
}];
以下为打印结果,从执行来看我们只执行了一个队列的信号的next事件,但其会将此队列中所有的signal信号都执行一遍(注意使用此种队列必须实现sendcompleted方法)
这里还有一种组队列的方式,其可以不要求signal实现sendcompleted方法,具体代码如下
//信号合并队列:当其中信号方法执行完后便会执行下个信号
[[RACSignal merge:@[signalB,signalC,signalD]] subscribeNext:^(id x) {
// code...
}];
3.信号的合并
//此处signal发送了三个next事件,若signalA发送next事件,则触发三次next事件
RACSignal *signalA = [RACSignal createSignal:^RACDisposable *(id subscriber) {
[subscriber sendNext:@"我想你"];
[subscriber sendNext:@"我不想你"];
[subscriber sendNext:@"Test"];
return nil;
}];
RACSignal *signalB = [RACSignal createSignal:^RACDisposable *(id subscriber) {
[subscriber sendNext:@"嗯"];
[subscriber sendNext:@"你豁我"];
return nil;
}];
//合并signalA和signalB
[[RACSignal combineLatest:@[signalA, signalB]] subscribeNext:^(id x) {
NSLog(@"%@",x);
}];
打印结果如下:从结果可以看出此种合并会将第一个信号中最后一个sendnext与后面信号的所有sendnext结合起来作为一个数组,而next触发次数以signalB中的next次数为主
4.信号的压缩
RACSignal *signalA = [RACSignal createSignal:^RACDisposable *(id subscriber) {
[subscriber sendNext:@"我想你"];
[subscriber sendNext:@"我不想你"];
[subscriber sendNext:@"Test"];
return nil;
}];
RACSignal *signalB = [RACSignal createSignal:^RACDisposable *(id subscriber) {
[subscriber sendNext:@"嗯"];
[subscriber sendNext:@"你豁我"];
return nil;
}];
// 压缩具有一一对应关系,以2个信号中 消息发送数量少的为主对应
[[signalA zipWith:signalB] subscribeNext:^(RACTuple* x) {
//解包RACTuple中的对象
RACTupleUnpack(NSString *stringA, NSString *stringB) = x;
NSLog(@"%@%@", stringA, stringB);
}];
打印结果如下:若将此结果于合并作对比,我们可以发现他们只是触发next事件的次数所关联对象不一样,是以信号中next事件数量较少的为主
5.秩序(同合并组队列相似)
[[[[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
NSLog(@"第一步");
[subscriber sendCompleted];
return nil;
}] then:^RACSignal *{
return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
NSLog(@"第二步");
[subscriber sendCompleted];
return nil;
}];
}] then:^RACSignal *{
return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
NSLog(@"第三步");
return nil;
}];
}] subscribeCompleted:^{
NSLog(@"完成");
}];
打印结果如下:
6.信号定时启动与超时处理
//设置定时启用,类似于NSTimer,这里需设置take方式
[[[RACSignal interval: onScheduler:[RACScheduler mainThreadScheduler]] take:]subscribeNext:^(id x) {
NSLog(@"吃药");
}];
//超时操作
[[[RACSignal createSignal:^RACDisposable *(id subscriber) {
[[[RACSignal createSignal:^RACDisposable *(id subscriber) {
NSLog(@"我快到了");
[subscriber sendNext:nil];
[subscriber sendCompleted];
return nil;
//延迟2秒后执行next事件
}] delay:] subscribeNext:^(id x) {
NSLog(@"我到了");
[subscriber sendNext:nil];
[subscriber sendCompleted];
}];
return nil;
}] timeout: onScheduler:[RACScheduler mainThreadScheduler]] subscribeError:^(NSError *error) {
NSLog(@"你再不来,我走了");
}];
打印结果如下:这里读者可以通过调试delay和timeout的数值来测试其使用方式。
7.信号设置error后重新执行
__block int i = ;
[[[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) { NSLog(@"i = %d",i);
if (i == ) {
[subscriber sendNext:@"i == 2"];
}else{
i ++;
[subscriber sendError:nil];
}
return nil;
//当发送的是error时可以retry重新执行
}] retry] subscribeNext:^(id x) {
NSLog(@"%@",x);
}];
打印结果如下:若发送的是error则可以使用retry来尝试重新刺激信号
8.信号刺激的条件设置
//创建一个信号
[[[RACSignal createSignal:^RACDisposable *(id subscriber) {
//创建一个定时信号,每隔1秒刺激一次信号
[[RACSignal interval: onScheduler:[RACScheduler mainThreadScheduler]] subscribeNext:^(id x) {
[subscriber sendNext:@"直到世界的尽头才能把我们分开"];
}];
return nil;
//直到此情况下停止刺激信号
}] takeUntil:[RACSignal createSignal:^RACDisposable *(id subscriber) {
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)( * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
NSLog(@"世界的尽头到了");
[subscriber sendNext:@"世界的尽头到了"];
});
return nil;
}]] subscribeNext:^(id x) {
NSLog(@"%@", x);
}];
打印结果如下:这样当某个条件达到后,就可以停止定时器了
以上都属于我从文章中提取出的个人觉得有帮助的元素,其文章参考地址为:http://www.cocoachina.com/ios/20150817/13071.html。若上文中有何错误使用或理解错误的地方望读者指出。谢谢!
ReactiveCocoa信号使用方法的更多相关文章
- 项目中学习ReactiveCocoa的使用方法
一.注册控制器 控制器上的一个属性 @property (weak, nonatomic) IBOutlet UIBarButtonItem *signInBtn; 在 viewDidLoad 方法中 ...
- ReactiveCocoa的使用方法
http://www.open-open.com/lib/view/open1440060663129.html best praticse https://github.com/ReactiveCo ...
- 支持5G-WiFi的安卓设备搜索不到5G信号解决方法
安卓设备必须获得root权限,然后修改 /system/etc/wifi/nvram_net.txt 文件, 将ccode = CN 改为 ccode = ALL.保存并重启即可. 三星EK-GC11 ...
- ReactiveCocoa的冷信号与热信号 探讨
背景 ReactiveCocoa(简称RAC)是最初由GitHub团队开发的一套基于Cocoa的FRP框架.FRP即Functional Reactive Programming(函数式响应式编程), ...
- ReactiveCocoa 中 RACSignal 是怎样发送信号
前言 ReactiveCocoa是一个(第一个?)将函数响应式编程范例带入Objective-C的开源库.ReactiveCocoa是由Josh Abernathy和Justin Spahr-Summ ...
- ReactiveCocoa 中 RACSignal 是如何发送信号的
https://juejin.im/post/5829f4c3570c350063c436ac 前言 ReactiveCocoa是一个(第一个?)将函数响应式编程范例带入Objective-C的开源库 ...
- QML 信号与响应方法的总结
以下内容为本人的著作,如需要转载,请声明原文链接微信公众号「englyf」https://www.cnblogs.com/englyf/p/16748191.html 如果面试过程中,面试官想了解你对 ...
- iOS开发之ReactiveCocoa下的MVVM(干货分享)
最近工作比较忙,但还是出来更新博客了,今天给大家分享一些ReactiveCocoa以及MVVM的一些东西,干活还是比较足的.在之前发表过一篇博文,名字叫做<iOS开发之浅谈MVVM的架构设计与团 ...
- ReactiveCocoa源码拆分解析(五)
(整个关于ReactiveCocoa的代码工程可以在https://github.com/qianhongqiang/QHQReactive下载) 好多天没写东西了,今天继续.主要讲解RAC如何于UI ...
随机推荐
- DevExpress GridView加入DevExpress中的右键菜单PopuMenu
1. 添加一个Barmanager控件 2. 加入popumenu控件,点击该控件右上角的黑色三角号,编辑选项,点击编辑的选项,选择事件,编辑事件. 3. 在使用该右键菜单的控件添加MouseUp事件 ...
- Airline Hub
参考:http://blog.csdn.net/mobius_strip/article/details/12731459 #include <stdio.h> #include < ...
- Construct Binary Tree from Inorder and Postorder Traversal || LeetCode
/** * Definition for a binary tree node. * struct TreeNode { * int val; * struct TreeNode *left; * s ...
- Linux modules install
安装模块的时候出现错误:modprobe: chdir(3.0.35-g6774ed9-dirty): No such file or directory. 内核模块没有安装正确.本文记录解决方法. ...
- js注册验证提示!
<script> var ifEmail =false; var ifPassword; function ainf() { var txtEnun=document.getElement ...
- git如何使用 svn如何使用
git和svn是2款常用的版本控制系统. git 的功能: 1.从服务器上克隆完整的Git仓库(包括代码和版本信息)到单机上. 也就是说自己机器上有一个git仓库. 这和svn是不同的,svn是没有本 ...
- 【转】数据库系统异常排查之DMV
数据库系统异常是DBA经常要面临的情景,一名有一定从业经验的DBA,都会有自己一套故障排查的方法和步骤,此文为为大家介绍一下通过系统 性能视图(SQLServer05以上版本)来排查系统异常的基本方法 ...
- 用JS来控制 div的高度随浏览器变化而变化
<div id="test" style=" border: solid 1px #f00; "></div> <script t ...
- 说一下linux中shell的后台进程与前台进程
环境: 操作系统:archlinux; 终端模拟器:rxvt-unicode(urxvt); shell:bash; 这里所说的后台进程是指在命令行后面加一个 ampersand(&),前台进 ...
- zw版·全程图解Halcon控件安装(delphi2007版)
zw版·全程图解Halcon控件安装(delphi2007版) delphi+halcon,这个组合,可以说是图像分析的神级配置,无论是开发效率,还是运行实在是太高了,分分钟秒杀c+opencv,py ...