iOS 初探ReactiveCocoa(二)

2017-09-04  本文已影响17人  iOS小武哥

1.ReactiveCocoa简介

ReactiveCocoa作用

在我们iOS开发过程中,当某些事件响应的时候,需要处理某些业务逻辑,这些事件都用不同的方式来处理。
比如按钮的点击使用actionScrollView滚动使用delegate,属性值改变使用KVO等系统提供的方式。
其实这些事件,都可以通过RAC处理
ReactiveCocoa为事件提供了很多处理方法,而且利用RAC处理事件很方便,可以把要处理的事情,和监听的事情的代码放在一起,这样非常方便我们管理,就不需要跳到对应的方法里。非常符合我们开发中高聚合,低耦合
的思想。

编程思想

在开发中我们也不能太依赖于某个框架,否则这个框架不更新了,导致项目后期没办法维护,比如之前Facebook提供的Three20框架,在当时也是神器,但是后来不更新了,也就没什么人用了。因此我感觉学习一个框架,还是有必要了解它的编程思想。
先简单介绍下目前咱们已知的编程思想。

链式编程特点:

方法的返回值是block,block必须有返回值(本身对象),block参数(需要操作的值)

响应式编程思想:

不需要考虑调用顺序,只需要知道考虑结果,类似于蝴蝶效应,产生一个事件,会影响很多东西,这些事件像流一样的传播出去,然后影响结果,借用面向对象的一句话,万物皆是流。
代表:KVO运用。

函数式编程思想

代表:ReactiveCocoa

ReactiveCocoa编程思想

ReactiveCocoa结合了几种编程风格:

所以,你可能听说过ReactiveCocoa被描述为函数响应式编程(FRP)框架。
以后使用RAC解决问题,就不需要考虑调用顺序,直接考虑结果,把每一次操作都写成一系列嵌套的方法中,使代码高聚合,方便管理。
5.如何导入ReactiveCocoa框架
通常都会使用CocoaPods(用于管理第三方框架的插件)帮助我们导入。
PS:CocoaPods教程

需要在podfile加上use_frameworks,重新pod install 才能导入成功。

ReactiveCocoa常见类。

学习框架首要之处:个人认为先要搞清楚框架中常用的类,在RAC中最核心的类RACSiganl,搞定这个类就能用ReactiveCocoa开发了。
如何订阅信号:调用信号RACSignal的subscribeNext就能订阅。

RACSiganl简单使用:

 + (RACSignal *)createSignal:(RACDisposable * (^)(id<RACSubscriber> subscriber))didSubscribe 
 - (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock

RACSignal底层实现:

 RACSignal *siganl = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) { 
  block调用时刻:每当有订阅者订阅信号,就会调用block。
 发送信号:
 [subscriber sendNext:@1];
 如果不在发送数据,最好发送信号完成,内部会自动调用[RACDisposable disposable]取消订阅信号。
 [subscriber sendCompleted]; 
return [RACDisposable disposableWithBlock:^{ 
 block调用时刻:当信号发送完成或者发送错误,就会自动执行这个block,取消订阅信号。
 // 执行完Block后,当前信号就不在被订阅了。
 NSLog(@"信号被销毁"); }];
 }]; 
订阅信号,才会激活信号. 
[siganl subscribeNext:^(id x) { 
// block调用时刻:每当有信号发出数据,就会调用block. NSLog(@"接收到数据:%@",x); 
}];

RACSubscriber

表示订阅者的意思,用于发送信号,这是一个协议,不是一个类,只要遵守这个协议,并且实现方法才能成为订阅者。通过create创建的信号,都有一个订阅者,帮助他发送数据。

RACDisposable

RACSubject

RACReplaySubject

RACSubjectRACReplaySubject简单使用:

RACSubject使用步骤:

RACSubject:底层实现和RACSignal不一样。

RACSubject 使用:

RACSubject *subject = [RACSubject subject];

[subject subscribeNext:^(id x) { 
// block调用时刻:当信号发出新值,就会调用. 
NSLog(@"第一个订阅者%@",x);
 }]; 
[subject subscribeNext:^(id x) {
 // block调用时刻:当信号发出新值,就会调用. NSLog(@"第二个订阅者%@",x);
 }]; 

[subject sendNext:@"1"];

RACReplaySubject使用步骤:
 [RACSubject subject]; 跟RACSiganl不一样,创建信号时没有block
 - (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock
RACReplaySubject:底层实现和RACSubject不一样。
 RACReplaySubject *replaySubject = [RACReplaySubject subject];
[replaySubject sendNext:@1]; 
[replaySubject sendNext:@2];
 [replaySubject subscribeNext:^(id x) {
 NSLog(@"第一个订阅者接收到的数据%@",x);
 }];

// 订阅信号:

 [replaySubject subscribeNext:^(id x) { 
NSLog(@"第二个订阅者接收到的数据%@",x); 
}];

RACSubject替换代理

需求: 给当前控制器添加一个按钮,modal到另一个控制器界面,另一个控制器view中有个按钮,点击按钮,通知当前控制器.
@interface TwoViewController : UIViewController
@property (nonatomic, strong) RACSubject *delegateSignal;
@end
@implementation TwoViewController
- (IBAction)notice:(id)sender { 
// 通知第一个控制器,告诉它,按钮被点了 // 通知代理 // 判断代理信号是否有值 
if (self.delegateSignal) { 
// 有值,才需要通知
 [self.delegateSignal sendNext:nil];
 }
}

@end

@implementation OneViewController 
- (IBAction)btnClick:(id)sender { 
// 创建第二个控制器
 TwoViewController *twoVc = [[TwoViewController alloc] init];
 // 设置代理信号:
 twoVc.delegateSignal = [RACSubject subject];
 // 订阅代理信号:
 [twoVc.delegateSignal subscribeNext:^(id x) { 
  NSLog(@"点击了通知按钮"); 
}];
 // 跳转到第二个控制器 :
[self presentViewController:twoVc animated:YES completion:nil];
}
@end

RACTuple

RACSequence

使用场景
 // 这里其实是三步
 // 第一步: 把数组转换成集合RACSequence numbers.rac_sequence
 // 第二步: 把集合RACSequence转换RACSignal信号类,numbers.rac_sequence.signal
 // 第三步: 订阅信号,激活信号,会自动把集合中的所有值,遍历出来。
 [numbers.rac_sequence.signal subscribeNext:^(id x) { NSLog(@"%@",x); }]; 
NSDictionary *dict = @{@"name":@"xmg",@"age":@18}; [dict.rac_sequence.signal subscribeNext:^(RACTuple *x) {
 // 解包元组,会把元组的值,按顺序给参数里面的变量赋值 RACTupleUnpack(NSString *key,NSString *value) = x; 
// 相当于以下写法// NSString *key = x[0];
// NSString *value = x[1]; NSLog(@"%@ %@",key,value);
 }]; 
 NSString *filePath = [[NSBundle mainBundle] pathForResource:@"flags.plist" ofType:nil];
 NSArray *dictArr = [NSArray arrayWithContentsOfFile:filePath]; NSMutableArray *items = [NSMutableArray array]; 
for (NSDictionary *dict in dictArr) {
 FlagItem *item = [FlagItem flagWithDict:dict]; 
[items addObject:item];
 } 
 NSString *filePath = [[NSBundle mainBundle] pathForResource:@"flags.plist" ofType:nil]; 
NSArray *dictArr = [NSArray arrayWithContentsOfFile:filePath]; NSMutableArray *flags = [NSMutableArray array]; 
_flags = flags; 
// rac_sequence注意点:调用subscribeNext,并不会马上执行nextBlock,而是会等一会。
 [dictArr.rac_sequence.signal subscribeNext:^(id x) { 
// 运用RAC遍历字典,x:字典 
FlagItem *item = [FlagItem flagWithDict:x];
 [flags addObject:item];
 }];
 NSLog(@"%@", NSStringFromCGRect([UIScreen mainScreen].bounds)); 
RAC高级写法:
 NSString *filePath = [[NSBundle mainBundle] pathForResource:@"flags.plist" ofType:nil]; 
NSArray *dictArr = [NSArray arrayWithContentsOfFile:filePath];
 // map:映射的意思,目的:把原始值value映射成一个新值 
// array: 把集合转换成数组 
// 底层实现:当信号被订阅,会遍历集合中的原始值,映射成新值,并且保存到新的数组里。
 NSArray *flags = [[dictArr.rac_sequence map:^id(id value) { return [FlagItem flagWithDict:value];
 }] 
array];

RACCommand:RAC中用于处理事件的类,可以把事件如何处理,事件中的数据如何传递,包装到这个类中,他可以很方便的监控事件的执行过程。

initWithSignalBlock:(RACSignal * (^)(id input))signalBlock

  • (RACSignal *)execute:(id)input

监听当前命令是否正在执行executing

 RACCommand *command = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) { 
NSLog(@"执行命令");
 // 创建空信号,必须返回信号
  return [RACSignal empty]; 
// 4.2.创建信号,用来传递数据
 return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
 [subscriber sendNext:@"请求数据"];
 // 注意:数据传递完,最好调用sendCompleted,这时命令才执行完毕。 
[subscriber sendCompleted];
 return nil;
 }];
 }];

// 强引用命令,不要被销毁,否则接收不到数据
_conmmand = command;

 [command.executionSignals subscribeNext:^(id x) {
 [x subscribeNext:^(id x) { NSLog(@"%@",x);
 }];
 }]; 

RAC高级用法

switchToLatest:用于signal of signals,获取signal of signals发出的最新信号,也就是可以直接拿到RACCommand中的信号 :
[command.executionSignals.switchToLatest subscribeNext:^(id x) 
{ NSLog(@"%@",x); 
}]; 
 [[command.executing skip:1] subscribeNext:^(id x) { 
if ([x boolValue] == YES)
 { 
// 正在执行
 NSLog(@"正在执行");
 }else{
 // 执行完成 NSLog(@"执行完成"); 
} 
}]; 
RACMulticastConnection
RACMulticastConnection简单使用:
 + (RACSignal *)createSignal:(RACDisposable * (^)(id<RACSubscriber> subscriber))didSubscribe 

创建连接:

 RACMulticastConnection *connect = [signal publish]; 

订阅信号,注意:订阅的不在是之前的信号,而是连接的信号。 [connect.signal subscribeNext:nextBlock];

连接:
[connect connect]

RACMulticastConnection底层原理:
connect.sourceSignal -> RACSignal(原始信号) connect.signal -> RACSubject
需求:假设在一个信号中发送请求,每次订阅一次都会发送请求,这样就会导致多次请求。

解决:使用RACMulticastConnection就能解决.

 RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) { 
NSLog(@"发送请求"); 
return nil; 
}];

订阅信号:

 [signal subscribeNext:^(id x) {
 NSLog(@"接收数据"); }]; 
订阅信号:
 [signal subscribeNext:^(id x) {
 NSLog(@"接收数据");
 }]; 

运行结果,会执行两遍发送请求,也就是每次订阅都会发送一次请求:
RACMulticastConnection:解决重复请求问题

 RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) { 
NSLog(@"发送请求"); 
[subscriber sendNext:@1];
 return nil;
 }];

RACMulticastConnection *connect = [signal publish];

 [connect.signal subscribeNext:^(id x) { 
NSLog(@"订阅者一信号");
 }]; 
[connect.signal subscribeNext:^(id x) {
 NSLog(@"订阅者二信号");
 }]; 
- 连接,激活信号:
 [connect connect];
RACScheduler
RACUnit
RACEvent

ReactiveCocoa开发中常见用法。

代码演示:
 [[redV rac_signalForSelector:@selector(btnClick:)] subscribeNext:^(id x)
 {
 NSLog(@"点击红色按钮");
 }];
 [[redV rac_valuesAndChangesForKeyPath:@"center" options:NSKeyValueObservingOptionNew observer:nil] 
subscribeNext:^(id x) { 
NSLog(@"%@",x);
 }];
 [[self.btn rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
 NSLog(@"按钮被点击了");
 }];
 [[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIKeyboardWillShowNotification object:nil] subscribeNext:^(id x) {
 NSLog(@"键盘弹出");
 }]; 
 [_textField.rac_textSignal subscribeNext:^(id x) { 
NSLog(@"文字改变了%@",x);
 }]; 
RACSignal *request1 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) { 
// 发送请求1:
 [subscriber sendNext:@"发送请求1"]; 
return nil; }];
 RACSignal *request2 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
 // 发送请求2:
 [subscriber sendNext:@"发送请求2"];
 return nil;
 }];
使用注意:几个信号,参数一的方法就几个参数,每个参数对应信号发出的数据.
 [self rac_liftSelector:@selector(updateUIWithR1:r2:) withSignalsFromArray:@[request1,request2]];}
- (void)updateUIWithR1:(id)data r2:(id)data1{ 
NSLog(@"更新UI%@ %@",data,data1);
}

ReactiveCocoa常见宏。

RAC(TARGET, [KEYPATH, [NIL_VALUE]]):用于给某个对象的某个属性绑定。

[RACObserve(self.view, center) subscribeNext:^(id x) { NSLog(@"%@",x); }];
 name = @"xmg" age = @20 
RACTupleUnpack(NSString *name,NSNumber *age) = tuple;

原文链接

上一篇下一篇

猜你喜欢

热点阅读