函数响应式编程及ReactiveObjC学习笔记 (三)
之前讲了RAC如何帮我们实现KVO / 代理 / 事件 / 通知
今天先不去分析它的核心代码, 我们先看看ReactiveObjC库里面一些特别的东西, 如果大家点开ReactiveObjC目录应该会看到很多category, 今天我们先来看看这些

我们先从UITextView+RACSignalSupport.h开始看
#import <UIKit/UIKit.h> @class RACDelegateProxy;
@class RACSignal<__covariant ValueType>; NS_ASSUME_NONNULL_BEGIN @interface UITextView (RACSignalSupport) @property (nonatomic, strong, readonly) RACDelegateProxy *rac_delegateProxy; - (RACSignal<NSString *> *)rac_textSignal; @end
这里有一个属性跟一个方法,
关于RACDelegateProxy这个类的用途大概是把初始化传入的代理绑定或者添加给当前正在处理的信号
给大家一个例子:
#import "ViewController.h"
#import <ReactiveObjC.h>
#import <objc/runtime.h> @interface ViewController ()<UITextViewDelegate> @end @implementation ViewController - (void)viewDidLoad {;
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib. // 以UITextViewDelegate来初始化一个RACDelegateProxy
RACDelegateProxy *delegateProxy = [[RACDelegateProxy alloc] initWithProtocol:@protocol(UITextViewDelegate)]; // 注册要实现的方法
[[delegateProxy
rac_signalForSelector:@selector(textViewDidBeginEditing:)]
subscribeNext:^(RACTuple * _Nullable x) { NSLog(@"开始编辑");
}]; UITextView *textView = [[UITextView alloc] initWithFrame:CGRectMake(, , , )];
textView.center = self.view.center;
textView.backgroundColor = [UIColor greenColor]; // 设置代理为我们创建的RACDelegateProxy, 注意要转义不然会有警告
textView.delegate = (id<UITextViewDelegate>)delegateProxy; [self.view addSubview:textView]; // retain我们创建的delegateProxy, 避免被释放
objc_setAssociatedObject(textView, _cmd, delegateProxy, OBJC_ASSOCIATION_RETAIN_NONATOMIC); } @end
这个一般是RAC内部使用, 我们比较少用. 另外也只能处理没有返回值的代理方法
可以到UITextview+RACSignalSupport.m里面看看, 也是类似这样用的
- (RACDelegateProxy *)rac_delegateProxy {
RACDelegateProxy *proxy = objc_getAssociatedObject(self, _cmd);
if (proxy == nil) {
proxy = [[RACDelegateProxy alloc] initWithProtocol:@protocol(UITextViewDelegate)];
objc_setAssociatedObject(self, _cmd, proxy, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
return proxy;
}
好了, 现在我们来使用下这个UITextView类别唯一的方法
- (RACSignal<NSString *> *)rac_textSignal;
大家可以看到, 这个方法会返回一个信号 我们可以对他订阅, 试试看
UITextView *textView = [[UITextView alloc] initWithFrame:CGRectMake(, , , )];
textView.center = self.view.center;
textView.backgroundColor = [UIColor greenColor]; [self.view addSubview:textView]; [[textView
rac_textSignal]
subscribeNext:^(NSString * _Nullable x) { NSLog(@"%@", x);
}];
运行看看, 当我们在textView中输入文字的时候会打印:
-- ::42.841 RAC[:]
-- ::43.353 RAC[:]
-- ::44.031 RAC[:]
所以这个x就是Textview的内容了.
下面我们看看
UITextField+RACSignalSupport.h
#import <UIKit/UIKit.h> @class RACChannelTerminal<ValueType>;
@class RACSignal<__covariant ValueType>; NS_ASSUME_NONNULL_BEGIN @interface UITextField (RACSignalSupport) - (RACSignal<NSString *> *)rac_textSignal; - (RACChannelTerminal<NSString *> *)rac_newTextChannel; @end NS_ASSUME_NONNULL_END
这里有两个方法, 我们先看第一个rac_textSignal
UITextField *textField = [[UITextField alloc] initWithFrame:CGRectMake(, , , )];
textField.center = self.view.center;
textField.backgroundColor = [UIColor yellowColor];
[[textField
rac_textSignal]
subscribeNext:^(NSString * _Nullable x) {
NSLog(@"%@", x);
}];
[self.view addSubview:textField];
运行看看, 我们输入数字的时候会打印内容:
-- ::45.686 RAC[:]
-- ::46.139 RAC[:]
-- ::46.798 RAC[:]
然后我们看看另外一个方法
- (RACChannelTerminal<NSString *> *)rac_newTextChannel;
这里涉及到了一个类RACChannelTerminal, 我们点进去看看这个类
@interface RACChannelTerminal<ValueType> : RACSignal<ValueType> <RACSubscriber>
- (instancetype)init __attribute__((unavailable("Instantiate a RACChannel instead")));
// Redeclaration of the RACSubscriber method. Made in order to specify a generic type.
- (void)sendNext:(nullable ValueType)value;
@end
可以看到它是一个RACSignal的子类, 我们先调用看看这个方法
[[textField
rac_newTextChannel]
subscribeNext:^(NSString * _Nullable x) { NSLog(@"%@", x);
}];
效果跟rac_textSignal一样, 那么它有什么特别的用法呢
它的作用是做双向绑定 关于什么是双向绑定呢? 给大家一个简单的例子:
UITextField *textFieldA = [[UITextField alloc] initWithFrame:CGRectMake(, , , )];
textFieldA.center = self.view.center;
textFieldA.backgroundColor = [UIColor yellowColor];
[self.view addSubview:textFieldA];
UITextField *textFieldB = [[UITextField alloc] initWithFrame:CGRectMake(, , , )];
textFieldB.center = CGPointMake(self.view.center.x, self.view.center.y + );
textFieldB.backgroundColor = [UIColor yellowColor];
[self.view addSubview:textFieldB];
RACChannelTerminal *terminalA = [textFieldA rac_newTextChannel];
RACChannelTerminal *terminalB = [textFieldB rac_newTextChannel];
[terminalA subscribe:terminalB];
[terminalB subscribe:terminalA];
运行可以看到, 改变textFieldA的值textFieldB的值也会跟着改变, 反过来也一样.
这里如果要实现双向绑定, 其实还有一个简单的方法:
RACChannelTo(textFieldA, text) = RACChannelTo(textFieldB, text);
大家可以试试看.
如果我们不仅仅想让两个绑定对象之间的值简单的相等而已呢? 比如textFieldA的值是123的时候textFieldB的值要为321要怎么处理呢?
这里我们先说一个一会用到的方法: map
map方法,将会创建一个和原来一模一样的信号,只不过新的信号传递的值变为了block(value)。
[[[textField
rac_textSignal]
map:^id _Nullable(NSString * _Nullable value) { if ([value isEqualToString:@""]) { return @"";
} else { return @"";
}
}] subscribeNext:^(id _Nullable x) { NSLog(@"%@", x);
}];
运行看看, 当我们输入1, 会打印0, 输入11的时候会打印1, 这里就是把传递的值从textField的text转变成为我们的1 和 0;
然后有个特别的地方, 加入我们知道传递的值的类型, 我们就可以直接把后面订阅的block里面的参数类型直接改成我们知道的类型
例如把id改为NSString *运行结果也是一样的, 这个是RAC一个比较特别的地方
那么要实现上面的123 到 321可以这样写:
RACChannelTerminal *terminalA = [textFieldA rac_newTextChannel];
RACChannelTerminal *terminalB = [textFieldB rac_newTextChannel]; [[terminalA
map:^id _Nullable(id _Nullable value) { if ([value isEqualToString:@""]) { return @"";
} return value;
}]
subscribe:terminalB]; [terminalB subscribe:terminalA];
大家可以自己运行看看效果, 当textFieldA输入123的时候textFieldB会变为321
下面我们看看
UIActionSheet+RACSignalSupport.h
#import <UIKit/UIKit.h> @class RACDelegateProxy;
@class RACSignal<__covariant ValueType>; NS_ASSUME_NONNULL_BEGIN @interface UIActionSheet (RACSignalSupport) @property (nonatomic, strong, readonly) RACDelegateProxy *rac_delegateProxy; - (RACSignal<NSNumber *> *)rac_buttonClickedSignal; @end
rac_delegateProxy跟之前textview是一样的用法这里开始就不再解释这类属性了
我们直接试着使用rac_buttonClickedSignal
UIActionSheet *actionSheet = [[UIActionSheet alloc] initWithTitle:@"RAC ActionSheet" delegate:self cancelButtonTitle:@"Cancel" destructiveButtonTitle:@"OK" otherButtonTitles:@"Other", nil];
[[actionSheet
rac_buttonClickedSignal]
subscribeNext:^(NSNumber * _Nullable x) {
NSLog(@"%@", x);
}];
[actionSheet showInView:self.view];
运行看看, x是actionSheet上按钮的编号, 我们拿到编号就可以做响应的事件处理了.
UIAlertView+RACSignalSupport.h
@interface UIAlertView (RACSignalSupport) @property (nonatomic, strong, readonly) RACDelegateProxy *rac_delegateProxy; - (RACSignal<NSNumber *> *)rac_buttonClickedSignal; - (RACSignal<NSNumber *> *)rac_willDismissSignal; @end NS_ASSUME_NONNULL_END
它有两个方法, 一个是点击的时候用, 一个是dismiss的时候用
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"RAC" message:@"RAC Alert" delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"OK", nil];
[[alert
rac_buttonClickedSignal]
subscribeNext:^(NSNumber * _Nullable x) {
NSLog(@"click: x");
}];
[[alert
rac_willDismissSignal]
subscribeNext:^(NSNumber * _Nullable x) {
NSLog(@"dismiss: %@", x);
}];
[alert show];
UIControl+RACSignalSupport.h
#import <UIKit/UIKit.h> @class RACSignal<__covariant ValueType>; NS_ASSUME_NONNULL_BEGIN @interface UIControl (RACSignalSupport) - (RACSignal<__kindof UIControl *> *)rac_signalForControlEvents:(UIControlEvents)controlEvents; @end NS_ASSUME_NONNULL_END
只有一个方法, 这个之前讲过是做UIControllerEvent处理的, 再给个例子:
UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
[button setFrame:CGRectMake(, , , )];
[button setCenter:self.view.center];
[button setBackgroundColor:[UIColor yellowColor]];
[button setTitle:@"按钮" forState:UIControlStateNormal];
[[button
rac_signalForControlEvents:UIControlEventTouchUpInside]
subscribeNext:^(__kindof UIControl * _Nullable x) {
NSLog(@"点击了按钮");
x.backgroundColor = [UIColor redColor];
}];
[self.view addSubview:button];
UIDatePicker+RACSignalSupport.h
#import <UIKit/UIKit.h> @class RACChannelTerminal<ValueType>; NS_ASSUME_NONNULL_BEGIN @interface UIDatePicker (RACSignalSupport) - (RACChannelTerminal<NSDate *> *)rac_newDateChannelWithNilValue:(nullable NSDate *)nilValue; @end NS_ASSUME_NONNULL_END
它只有一个绑定的方法, 直接给大家一个例子:
大概效果为我们在Controller中添加一个UITextField跟一个UIDatePicker, 然后获取他们的RACChannelTerminal,
将UIDatePicker绑定给UITextField, 当我们滚动datePicker的时候 textField的值会跟着改变
UIDatePicker *datePicker = [[UIDatePicker alloc] initWithFrame:CGRectMake(, , self.view.frame.size.width, )];
datePicker.center = self.view.center;
datePicker.backgroundColor = [UIColor redColor]; [self.view addSubview:datePicker]; UITextField *textField = [[UITextField alloc] initWithFrame:CGRectMake(, , , )];
textField.center = CGPointMake(self.view.center.x, self.view.center.y - );
textField.backgroundColor = [UIColor yellowColor]; [self.view addSubview:textField]; RACChannelTerminal *datePickerTerminal = [datePicker rac_newDateChannelWithNilValue:[NSDate date]];
RACChannelTerminal *textFieldTerminal = [textField rac_newTextChannel]; [[datePickerTerminal
map:^id _Nullable(id _Nullable value) { NSLog(@"%@", value);
NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
[dateFormatter setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
return [dateFormatter stringFromDate:value];
}]
subscribe:textFieldTerminal];
运行截图:

这篇先写到这里, 下次我们再接着看
函数响应式编程及ReactiveObjC学习笔记 (三)的更多相关文章
- 函数响应式编程及ReactiveObjC学习笔记 (-)
最近无意间看到一个视频讲的ReactiveObjC, 觉得挺好用的 但听完后只是了解个大概. 在网上找了些文章, 有的写的比较易懂但看完还是没觉得自己能比较好的使用RAC, 有的甚至让我看不下去 这两 ...
- 函数响应式编程及ReactiveObjC学习笔记 (二)
之前我们初步认识了RAC的设计思路跟实现方式, 现在我们再来看看如果使用它以及它能帮我们做什么 One of the major advantages of RAC is that it provid ...
- 函数响应式编程及ReactiveObjC学习笔记 (四)
今天我们继续看其他的类别 UIImagePickerController+RACSignalSupport.h #import <UIKit/UIKit.h> @class RACDele ...
- 函数响应式编程(FRP)框架--ReactiveCocoa
由于工作原因,有段时间没更新博客了,甚是抱歉,只是,从今天開始我又活跃起来了,哈哈,于是决定每周更新一博.大家互相学习.交流. 今天呢.讨论一下关于ReactiveCocoa,这个採用函数响应式编程( ...
- 函数响应式编程(FRP)—基础概念篇
原文出处:http://ios.jobbole.com/86815/. 一函数响应式编程 说到函数响应式编程,就不得不提到函数式编程,他们俩有什么关系呢?今天我们就详细的解析一下他们的关系. 现在下面 ...
- RxJS入门之函数响应式编程
一.函数式编程 1.声明式(Declarativ) 和声明式相对应的编程⽅式叫做命令式编程(ImperativeProgramming),命令式编程也是最常见的⼀种编程⽅式. //命令式编程: fun ...
- 函数响应式编程(FRP)思想-Callback风格
序 ReactiveCocoa是IOS广为使用的技术框架,而ReactiveCocoa的核心思想就FRP.FRP不同于JAVA的object-oriented和AOP,FRP能让你的代码像数学一样简洁 ...
- ReactiveCocoa,最受欢迎的iOS函数响应式编程库(2.5版),没有之一!
简介 项目主页: ReactiveCocoa 实例下载: https://github.com/ios122/ios122 简评: 最受欢迎,最有价值的iOS响应式编程库,没有之一!iOS MVVM模 ...
- 函数响应式编程(FRP)从入门到”放弃”——基础概念篇
前言 研究ReactiveCocoa一段时间了,是时候总结一下学到的一些知识了. 一.函数响应式编程 说道函数响应式编程,就不得不提到函数式编程,它们俩到底有什么关系呢?今天我们就详细的解析一下他们的 ...
随机推荐
- Webpack插件开发简要
背景 如今'大前端'这个概念在前端界大热,说'大前端',我们就要提到'前后端分离','前后端分离'又离不开'本地开发构建','本地开发构建'自然离不开webpack,webpack想要工作,那它就需要 ...
- asp.net MVC 网站图片防盗链的几种方法
目录 1. 通过 URL Rewrite Module 组件 2. 通过 nginx 图片防盗链 3.自定义 HttpHandler 处理 4. 通过 MVC 自定义路由规则防盗链 5. 通过 MVC ...
- 关于JQuery获取宽度和高度在chrome和IE下的不同
之前写了一个关于滚动条的东西,可是在写的时候发现JQuery在获取宽度和高度时在不同浏览器中是不一样的,下面发一下代码给给位看官先展示一下: $(function(){ $("#main&q ...
- cpp(第五章)
1.副作用,指的是在计算表达式时对某些东西(如存储在变量的值)进行修改:顺序点,是程序执行过程中的一个点,在这里,进入下一步之前将确保对所有的副作用 都进行评估.(分号就是一个顺序点).for exa ...
- 高CPU、数据库无法读写的真凶
有兴趣的同学可以参考如下系列文章,都是针对dump分析的实战和总结: Windbg DUMP分析(原创汇总) http://www.cnblogs.com/LoveOfPrince/p/6653341 ...
- 用 BeautifulSoup爬取58商品信息
最近对Python爬虫比较迷恋,看了些爬虫相关的教程 于是乎跟着一起爬取了58上面的一些商品信息,并存入到xlsx文件中,并通过xlsxwirter的方法给表格设置了一些格式.好了,直接贴代码吧~ # ...
- Mac OS X 安装后的简单设置
让Mac拥有类似apt-get的功能--安装Homebrew Homebrew是一个包管理器,用于在Mac上安装一些OS X没有的UNIX工具(比如著名的wget). 国内下载地址:http://ww ...
- [codeforces113D]Museum
D. Museum time limit per test: 2 seconds memory limit per test: 256 megabytes input: standard input ...
- 制作Ubuntu Kylin局域网源
国人参与开发的开源操作系统UbuntuKylin(http://www.ubuntukylin.com/)已经发布有一段时间了,一直想在单位的局域网内部用用,可惜离线安装比较麻烦,于是搜索了些如何制作 ...
- java io 流
Java流操作有关的类或接口: Java流类图结构: 流的概念和作用 流是一组有顺序的,有起点和终点的字节集合,是对数据传输的总称或抽象.即数据在两设备间的传输称为流,流的本质是数据传输,根据数据传输 ...