RACCommand 接口中的高阶信号
每一个RACCommand对象中都管理着多个信号,它在接口中暴露出的四个信号是这一节关注的重点:
executionSignals是RACCommand中最重要的信号;从类型来看,它是一个包含信号的信号,在每次执行-execute:方法时,最终都会向executionSignals中传入一个最新的信号。
http://draveness.me/raccommand.html
executing是一个表示当前是否有任务执行的信号,这个信号使用了在上一节中介绍的临时变量作为数据源:
enabled信号流表示当前的命令是否可以再次被执行,也就是-execute:方法能否可以成功执行新的任务
错误信号是RACCommand中比较简单的信号;为了保证RACCommand对此执行-execute:方法也可以继续运行,我们只能将所有的错误以其它的形式发送到errors信号中,防止向executionSignals发送错误信号后,executionSignals信号就会中止的问题。
RACCommand 的使用
RACCommand非常适合封装网络请求,我们可以使用下面的代码封装一个网络请求
RACCommand*command=[[RACCommand alloc]initWithSignalBlock:^RACSignal*_Nonnull(id _Nullable input){return[RACSignal createSignal:^RACDisposable*_Nullable(id_Nonnull subscriber){NSURL*url=[NSURL URLWithString:@"http://localhost:3000"];AFHTTPSessionManager*manager=[[AFHTTPSessionManager alloc]initWithBaseURL:url];NSString*URLString=[NSString stringWithFormat:@"/api/products/%@",input?:@1];NSURLSessionDataTask*task=[manager GET:URLString parameters:nil progress:nil success:^(NSURLSessionDataTask*_Nonnull task,id _Nullable responseObject){[subscriber sendNext:responseObject];[subscriber sendCompleted];}failure:^(NSURLSessionDataTask*_Nullable task,NSError*_Nonnull error){[subscriber sendError:error];}];return[RACDisposable disposableWithBlock:^{[task cancel];}];}];}];
上面的RACCommand对象可以通过-execute:方法执行,同时,订阅executionSignals以及errors来获取网络请求的结果。
[[command.executionSignals switchToLatest]subscribeNext:^(id _Nullable x){NSLog(@"%@",x);}];
[command.errors subscribeNext:^(NSError*_Nullable x){NSLog(@"%@",x);}];[command execute:@1];
向方法-execute:中传入了@1对象,从服务器中获取了id = 1的商品对象;当然,我们也可以传入不同的id来获取不同的模型,所有的网络请求以及 JSON 转换模型的逻辑都可以封装到这个RACCommand的 block 中,外界只是传入一个id,最后就从executionSignals信号中获取了开箱即用的对象。
总结
使用RACCommand能够优雅地将包含副作用的操作和与副作用无关的操作分隔起来;整个RACCommand相当于一个黑箱,从-execute:方法中获得输入,最后以向信号发送消息的方式,向订阅者推送结果。
这种执行任务的方式就像是一个函数,根据输入的不同,有着不同的输出,非常适合与 UI、网络操作的相关的任务,这也是RACCommand的设计的优雅之处。