iOS ReactiveCocoa框架的简单使用

2018-12-18  本文已影响0人  云霄_云霄

写在前面:本文章非本人原创,原创地址为:https://www.jianshu.com/p/148075efc2c9

ReactiveCocoa框架的使用教程在网上有很多详细的博客可参考,通过学习,我自己也整理了一下,一来便于自己复习,二来分享给大家。先粘贴些优质博文的链接,然后下面以实例的形式一步步讲解。

下面开始介绍ReactiveCocoa的使用

一、在项目中集成ReactiveCocoa框架

既然是第三方框架,那用CocoaPods集成是最方便的。

首先,创建一个工程

然后,在工程中创建Podfile文件,文件中的内容如下:

platform :ios,'9.0'

use_frameworks!

target 'RWReactivePlayground' do

pod 'ReactiveCocoa', '~> 2.5'

end

注意1:

集成ReactiveCocoa框架和其他的不同之处是多了一个“use_frameworks!”,我在使用过程中发现,2.5版本以上的更高的版本要加上“use_frameworks!”,否则会报错,导致集成不了。而2.5版本之前的(包括2.5版本),就不需要加“use_frameworks!”,

注意2:

ReactiveCocoa现在的最高版本已经到5.0了,问题是,如果用swift编程,那么集成最新版本的ReactiveCocoa框架没有问题,但是如果使用OC编程的话,那最高只能集成2.5版本的RAC(RAC是ReactiveCocoa的简称),否则集成好了以后工程会报错。

简单的说就是,如果你用swift编程,用Cocoapods集成时,Podfile文件这么写

platform :ios,'9.0'

use_frameworks!

target 'RWReactivePlayground' do

pod 'ReactiveCocoa', '~> 5.0'

end

如果你用的是oc编程,用Cocoapods集成时,Podfile文件这么写

platform :ios,'9.0'

target 'RWReactivePlayground' do

pod 'ReactiveCocoa', '~> 2.5'

end

最后,上面的工作都做好了,就可以集成RAC了,很快.

屏幕快照 2017-02-27 上午10.59.13.png

二、RAC的简单使用----RACSignal

在要使用的RAC的控制器中导入RAC框架的头文件

#import<ReactiveCocoa/ReactiveCocoa.h>

现在来熟悉下RACSignal的使用,从名字就可以看出,它是信号。在viewDidload中加入下面的代码

//创建信号RACSignal * single = [RACSignal createSignal:^RACDisposable *(id subscriber) {NSLog(@"想");        [subscriber sendNext:@"发送了信号"];//发送信号NSLog(@"你");        [subscriber sendCompleted];//发送完成,订阅自动移除//RACDisposable 可用于手动移除订阅return[RACDisposable disposableWithBlock:^{NSLog(@"豆腐");        }];    }];//订阅信号NSLog(@"我");    [single subscribeNext:^(idx) {NSLog(@"吃");//        NSLog(@"信号的值:%@",x);}];

运行,得到结果如下

RACSinale.png

这样就可以清楚的看明白,信号的运行流程,但是感觉好乱,下面分析一下:

1.createSignal方法 是创建信号,创建好的信号,没有被订阅前,只是冷信号,此时是不会走createSignal后面的block的。

程序往下,就走到“NSLog(@"我")”,

2.然后走到subscribeNext,这一步就是订阅信号,订阅号信号后,信号single就变成了热信号,

3.既然变成热信号,就开始走createSignal后面的block中的去,所以就打印出了“NSLog(@"想")”。

4.下面是sendNext,即发送信号,发送了信号,订阅者就会收到信号,发送的内容可以从订阅信号subscribeNext后面的block中获取到,程序就走到subscribeNext后面的block中,所以就打印了“NSLog(@"吃")”,

5.当订阅信号的subscribeNext后面的block走完以后,程序又回到,createSignal后面的block中,继续未完成的代码,所以就打印“NSLog(@"你")”,继续往下就是[subscriber sendCompleted],这句代码的意思是,发送完成了,订阅自动移除,没有了订阅者了,信号又变成了冷信号。

6.接下来就是return,返回一个RACDisposable对象,这个的作用就是,可以用来手动移除订阅。RACDisposable对象,创建完成,就走进创建方法的block中,也就是打印NSLog(@"豆腐")

综上,打印出来的结果就是“我想吃你豆腐”,它就是这样出来的

这里再介绍下RACDisposable的使用,将代码改一下

//创建信号RACSignal * single = [RACSignal createSignal:^RACDisposable *(id subscriber) {NSLog(@"想");        [subscriber sendNext:@"发送了一个信号"];//发送信号NSLog(@"你");//RACDisposable 手动移除订阅者return[RACDisposable disposableWithBlock:^{NSLog(@"豆腐");        }];    }];//订阅信号NSLog(@"我");    RACDisposable * disposable = [single subscribeNext:^(idx) {NSLog(@"吃");NSLog(@"信号的值:%@",x);    }];//手动移除订阅[disposable dispose];

打印结果如下

在稍微分析一下,两份代码不同之处是,删去了自动移除订阅[subscriber sendCompleted],添加了手动删除订阅[disposable dispose],手动删除订阅,可以在你想要的地方,合适的时候进行操作。不过手动删除用的少。那既然用得少,我们还是用自动删除吧,优化下,见代码

//创建信号RACSignal * single = [RACSignal createSignal:^RACDisposable *(id subscriber) {NSLog(@"想");        [subscriber sendNext:@"发送了信号"];//发送信号NSLog(@"你");        [subscriber sendCompleted];//发送完成,订阅自动移除//RACDisposable 手动移除订阅者returnnil;    }];//订阅信号NSLog(@"我");    [single subscribeNext:^(idx) {NSLog(@"吃");NSLog(@"信号的值:%@",x);    }];

打印结果如下

好了,没豆腐吃了!其实也不需要。返回nil就可以了

上面罗里吧嗦的说了那么多,就是为了理清里面的逻辑,没有结合实际使用,其实听起来还是很迷糊,下面就结合实际,来使用RAC

第二、RAC的常用方法

上面是使用RACSignal创建信号,其实文本框中文字改变也是信号,按钮点击也是信号,RAC为UITxtField和UIButton创建categary,并做好了封装,直接就可以调用它们的信号,这里就围绕着这两个类,进行ARC的使用讲解

在工程中新建一个控制器,添加几个控件,textField,textView,button,label,如下图

订阅textField信号

[self.textfield.rac_textSignal subscribeNext:^(idx) {NSLog(@"%@",x);    }];

因为textField的信号肯定是NSString,类型的,所以可以写成下面的样子,也更方便使用些

[self.textfield.rac_textSignal subscribeNext:^(NSString* x) {NSLog(@"%@",x);    }];

这样,当你在文本框输入时,控制台就会打印出输入的内容,如下

可以看到,每次输入都会获取到信号。

filter---信号过滤器

如果我只需要将字符串长度超过3的,才打印,那可以使用过滤器filter,使用方法如下

[[self.textfield.rac_textSignal filter:^BOOL(NSString* value) {returnvalue.length>3?YES:NO;    }]subscribeNext:^(NSString* x) {NSLog(@"过滤后的到的信号:%@",x);    }];

在文本框中输入字符,打印结果如下

可以看到,只有当字符串长度大于3的信号,才会被订阅到

map--转换器

map就是将一种信号转换成你想要的另一种信号,这里把字符串信号,转换成文字信号

如果想当文本框中输入的文字长度大于4的时候,改变文本框的背景色,一种方法是把过滤器的条件设置为4,然后在subscribeNext的block中直接给textField.backgroundColor赋值。不过RAC有转换信号的方法---map,如下

[[[self.textfield.rac_textSignal filter:^BOOL(NSString* value) {returnvalue.length>3?YES:NO;    }]map:^id(NSString* value) {returnvalue.length >4?[UIColorredColor]:[UIColorwhiteColor];    }]subscribeNext:^(UIColor* value) {self.textfield.backgroundColor = value;    }];

上面的代码的意思是,当输入的字符串长度超过3,就将字符串信号转换成颜色信号,然后订阅该颜色信号,并将颜色赋值给textField的背景色。效果如下

这样的话,当字符串大于3,文本框的背景色变成了红色

另外,RAC提供了一个宏"RAC(对象,属性)"来简化代码并增强可读性,如下

RAC(self.textfield ,backgroundColor) = [self.textfield.rac_textSignal map:^id(NSString* value) {returnvalue.length >4?[UIColorredColor]:[UIColorwhiteColor];    }];

RAC宏有两个参数,一个是需要设置的对象,一个是设置的属性。这句代码的意思是,当文本框输入的字符串长度大于4时,改变文本框的背景色。这样的话,看起来更清晰,而达到的效果是一样的。

总结一下,到现在为止,学了过滤器:filter,转换器:map,对象设置属性的宏:RAC(要设置的对象,要设置的属性)。可以想象,用这几个方法可以很方便的实现一些功能,比喻说替代通知,监听事件等。

textField的使用是这样,那textView的使用也是这样的,因为他们完全类似

RAC(self.textView ,backgroundColor) = [self.textView.rac_textSignal map:^id(NSString* value) {returnvalue.length >4?[UIColorredColor]:[UIColorwhiteColor];    }];

combineLatest:reduce:

想象一下,如果当textField和textView同时满足某个条件时,才能进行某项操作的话,应该如何写呢?RAC为我们准备了一个方法--combineLatest:reduce:信号合并

先看代码

RACSignal * mergeTwoSignal = [RACSignal combineLatest:@[self.textfield.rac_textSignal,self.textView.rac_textSignal] reduce:^id(NSString* value1,NSString* value2){return[NSNumbernumberWithBool:([value1 isEqualToString:@"11111"]&&[value2 isEqualToString:@"22222"])];    }];RAC(self.addButton,enabled) = [mergeTwoSignal map:^id(NSNumber* value) {returnvalue;    }];

上面的代码的意思是,当textField中的文字为"11111",同时textView中的文字为"22222"的时候,返回一个信号,信号的类型是NSNumber,然后通过转换器map,将值返回,返回的值用于确定按钮是否可用。

可能会疑问,map中返回的NSNumber类型的,而button的enabled属性是BOOL类型,怎么可以这样直接赋值,但是RAC它就是可以,就是做的这么好。

到这一步,就可以订阅button的点击信号了,看代码就懂了

[[self.addButton rac_signalForControlEvents:(UIControlEventTouchUpInside)]    subscribeNext:^(NSNumber* value) {//标签赋值self.displayLabel.text =@"1314";    }];

运行验证一下,结果如下

确实能达到要求。真好,再也不用给button 添加点击事件了。

doNext

现在讲一下附加操作doNext,它的作用是,在不改变信号的基础上,进行一些附加的操作,比喻说,我在订阅到给label赋值前,改变label的背景色,当然也可以是做别的操作。反正是附加的不会影响信号流的。使用见代码

[[[self.addButton rac_signalForControlEvents:(UIControlEventTouchUpInside)]    doNext:^(idx) {//改变label的背景色self.displayLabel.backgroundColor = [UIColorredColor];    }]    subscribeNext:^(NSNumber* value) {self.displayLabel.text =@"1314";    }];

这样就实现了,订阅信号前,改变label的背景色

@weakify和@strongify

RAC的所有方法中,大部分是block,所以无法避免在使用过程中导致循环引用,

以前的解决办法是这样的

__weakSecondViewController *bself =self;    [[[self.addButton rac_signalForControlEvents:(UIControlEventTouchUpInside)]        doNext:^(idx) {//先清掉label中的文字bself.displayLabel.textColor = [UIColorredColor];        }]      subscribeNext:^(NSNumber* value) {          bself.displayLabel.text =@"1314";      }];

如果每个block都写的话,会很费劲,因为block太多了,还好RAC提供了两个宏,@weakify和@strongify,

@weakify(self);    [[[self.addButton rac_signalForControlEvents:(UIControlEventTouchUpInside)]      doNext:^(idx) {          @strongify(self);self.displayLabel.textColor = [UIColorredColor];      }]    subscribeNext:^(NSNumber* value) {        @strongify(self);self.displayLabel.text =@"1314";    }];

@weakify宏让你创建一个弱引用的影子对象(如果你需要多个弱引用,你可以传入多个变量),@strongify让你创建一个对之前传入@weakify对象的强引用。这样就解决了循环引用的问题

第三、RAC在网络请求和图片加载中的使用

先创建一个控制器,添加若干控件,textView,用来展示请求到的数据,imageView,用来展示图片,

使用系统的方法请求数据

在viewDidload中添加下面的代码

NSURL* url = [NSURLURLWithString:urlS];NSURLSession* session = [NSURLSessionsharedSession];NSMutableURLRequest* request = [NSMutableURLRequestrequestWithURL:url];NSURLSessionTask* task = [session dataTaskWithRequest:request completionHandler:^(NSData* _Nullable data,NSURLResponse* _Nullable response,NSError* _Nullable error) {NSString* dataString = [[NSStringalloc] initWithData:data encoding:NSUTF8StringEncoding];NSLog(@"%@",dataString);NSDictionary* dic = [NSJSONSerializationJSONObjectWithData:data options:0error:nil];NSLog(@"%@",dic);        [selfperformSelector:@selector(actionWithString:) onThread:[NSThreadmainThread] withObject:dataString waitUntilDone:YES];    }];    [task resume];

//回到主线程给textView赋值-(void)actionWithString:(id)value{self.textView.text = (NSString*)value;}

结果如下

使用RAC请求网络数据

把系统请求网络数据的方法,封装成信号流

//rac网络请求-(RACSignal *)racNetworkRequest{return[RACSignal createSignal:^RACDisposable *(id subscriber) {NSURL* url = [NSURLURLWithString:urlS];NSURLSession* session = [NSURLSessionsharedSession];NSMutableURLRequest* request = [NSMutableURLRequestrequestWithURL:url];NSURLSessionTask* task = [session dataTaskWithRequest:request completionHandler:^(NSData* _Nullable data,NSURLResponse* _Nullable response,NSError* _Nullable error) {NSString* dataString = [[NSStringalloc] initWithData:data encoding:NSUTF8StringEncoding];//            NSLog(@"%@",dataString);//            NSDictionary * dic = [NSJSONSerialization JSONObjectWithData:data options:0 error:nil];//            NSLog(@"%@",dic);if(error ==nil) {//返回成功[subscriber sendNext:dataString];//发送信号[subscriber sendCompleted];//结束发送}else{                [subscriber sendError:error];//发送错误}        }];        [task resume];returnnil;    }];}

现在就来调用一下看看,在viewDidLoad中添加下面的代码,

self.requestDataButton是一个按钮,使用方法是点击按钮的时候加载数据

[[[self.requestDataButton rac_signalForControlEvents:(UIControlEventTouchUpInside)]    map:^id(idvalue) {return[selfracNetworkRequest];    }]    subscribeNext:^(idx) {NSLog(@"%@",x);    }];

订阅信号后,得到的数据如下

发现,得到的不是想要的数据,而是一个信号对象,其实从racNetworkRequest这个方法中就可以看出,返回就是一个RACSignal对象,如果能获取到RACSignal对象里面的信号流就对了,怎么办呢,RAC提供了这样的方法 flattenMap

flattenMap---获取信号中的信号

把上面的代码写成这样,就可以获取到数据了

[[[self.requestDataButton rac_signalForControlEvents:(UIControlEventTouchUpInside)]      flattenMap:^id(idvalue) {return[selfracNetworkRequest];      }]    subscribeNext:^(idx) {NSLog(@"%@",x);    } error:^(NSError*error) {NSLog(@"%@",error);    }];

这样就会发现订阅到的信号,是你想要的数据了。

then---等待上一个信号的完成,然后订阅自己的信号

[[[selfracNetworkRequest]    then:^RACSignal *{returnself.textField.rac_textSignal;    }]    subscribeNext:^(idx) {NSLog(@"%@",x);    }error:^(NSError*error) {NSLog(@"error");    }];

then方法会等待前面的信号中completed事件的发送完成,然后再订阅由then block返回的信号。这样就高效地把控制权从一个signal传递给下一个。如此就实现了:当请求数据完成,就可以监控到textField中的文字输入了

回到主线程---deliverOn

因为信号的流转及操作都是在block中完成的,也就是说大部分操作都是在子线程中执行的操作,但是有个时候需要回到主线程完成一些事情,比如,请求到数据后,要刷新UI,这就必须回到主线程,RAC提供了这样的方法deliverOn。用法见下面的代码

@weakify(self)    [[[[[selfracNetworkRequest]        then:^RACSignal *{            @strongify(self);returnself.textField.rac_textSignal;        }]      filter:^BOOL(NSString* value) {returnvalue.length >3?YES:NO;      }]    deliverOn:[RACScheduler mainThreadScheduler]]//回到主线程subscribeNext:^(NSString* value) {        @strongify(self);self.textView.text =value;NSLog(@"%@",value);NSLog(@"当前线程%@",[NSThreadcurrentThread]);    } error:^(NSError*error) {NSLog(@"%@",error);    }];

这样的话,实现的效果就是,在textField中输入文字而且当文字大于3的时候,会在textView中显示出来,而且可以看到订阅信号的block中打印出来的线程是主线程,如下:

悲催的是,如果没有加deliverOn:好像也是在主线程。我也不知道什么原因,不知道有没有用,姑且就认为deliverOn有用,可能在开启很多线程的时候会有用吧

不过我可以在subscribeNext的block中加入回到主线程的方法,也能达到目的,如下

subscribeNext:^(NSString* value) {        @strongify(self);self.textView.text =value;NSLog(@"%@",value);        [selfperformSelectorOnMainThread:@selector(doSomething) withObject:nilwaitUntilDone:YES];NSLog(@"当前线程%@",[NSThreadcurrentThread]);    } error:^(NSError*error) {NSLog(@"%@",error);    }];

信号节流---throttle

用文本框textField作比喻,当我在里面输入字符时,subscribeNext的block会不停的走,每输入一个字符,就会走一遍。如果我想在输入过程中不需要每改变一个字符就走一遍,而是等输入完成或停止的时候再走block里面的代码,那就可以用throttle,先看效果

[[[self.textField.rac_textSignal      filter:^BOOL(NSString* value) {returnvalue.length >3?YES:NO;      }]      throttle:1]    subscribeNext:^(NSString* value) {        @strongify(self);self.textView.text =value;    } error:^(NSError*error) {NSLog(@"%@",error);    }];

这样得到的效果是,当输入字符串长度大于3,而且该字符串的值在1s内没有改变,就把textField中的值,赋值给textView。所以简单的说throttle的作用:如果前面信号在设定的时间内没有变化时,throttle就会把信号传到下面的事件中去。

使用系统的方法加载图片

系统的方法,我就不说了 看代码

dispatch_queue_tqueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT,0);dispatch_async(queue, ^{NSURL* url = [NSURLURLWithString:imageUrlString];UIImage* image = [UIImageimageWithData:[NSDatadataWithContentsOfURL:url]];if(image!=nil) {dispatch_async(dispatch_get_main_queue(), ^{self.imageView.image = image;            });        }    });

这样就可以完成图片的加载,看下面的效果

RAC加载图片

创建一个加载图片的方法,方法返回的是RACSignle信号对象,

-(RACSignal*)racRequestImage{return[RACSignal createSignal:^RACDisposable *(id subscriber) {NSURL* url = [NSURLURLWithString:imageUrlString];UIImage* image = [UIImageimageWithData:[NSDatadataWithContentsOfURL:url]];self.imageView.image = image;        [subscriber sendNext:image];        [subscriber sendCompleted];returnnil;    }];}

下面就来调用这个方法。实现的效果是,点击按钮(self.requestImageDataButton),即开始加载图片,在viewDidLoad中添加下面的代码

@weakify(self);    [[[self.requestImageDataButton rac_signalForControlEvents:(UIControlEventTouchUpInside)]      flattenMap:^RACStream *(idvalue) {return[selfracRequestImage];      }]    deliverOn:[RACScheduler mainThreadScheduler]]    subscribeNext:^(UIImage* image) {        @strongify(self);self.imageView.image = image;    }];

运行一下,发现,图片加载正常。从代码量来看,GCD可能方便一点,但是,如果是多个事件凑到一起影响图片加载的时候,RAC或许是不错的选择。

到这一步,就把ReactiveCocoa的初步使用讲完了。

总结一下总共学习哪些方法

filter---信号过滤器

map--转换器

combineLatest:reduce:--信号合并

doNext--附加操作

@weakify和@strongify--避免循环

flattenMap---获取信号中的信号

then---等待上一个信号的完成,然后订阅自己的信号

回到主线程---deliverOn

信号节流---throttle

作者:爬树的蚂蚁

链接:https://www.jianshu.com/p/148075efc2c9

來源:简书

简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。

上一篇 下一篇

猜你喜欢

热点阅读