ReactiveCocoa2

RACCommand

理解:个人觉得是对RACSignal的封装,侧重于对事件的信号的封装。有了RACComand,就可以把动作通过它很方便的放到ViewModel中,类似于以下这种方式:

1.在ViewModel中定义RACCommand

@interface SubscribeViewModel : NSObject
@property(nonatomic, strong) RACCommand *subscribeCommand;
 
@end

2.在ViewModel中处理具体的RACCommand封装的动作信号

 - (RACCommand *)subscribeCommand {
  if (!_subscribeCommand) {
      NSString *email = self.email;
      _subscribeCommand = [[RACCommand alloc] initWithEnabled:self.
      emailValidSignal signalBlock:^RACSignal *(id input) {
          return [SubscribeViewModel postEmail:email];
      }];
  }
  return _subscribeCommand;
}

input参数是在command调用excute方法的时候传进来的。

3.在ViewController中将ViewModel的RACCommand绑定到控件的事件信号上

self.subscribeButton.rac_command = self.viewModel.subscribeCommand;

如果要对某个行为进行单元测试也非常方便,直接对ViewModel层进行测试就可以了,并且没有任务Uiew层的显示逻辑,纯业务逻辑测起来很方便。如果没有RACCommand,直接通过signal去处理,可能类似这种方式:

[[self.button rac_signalForControlEvents:UIControlEventTouchUpInside]
subscribeNext:^(id x) {
}];

直接在ViewController中定义了动作的逻辑,没有很好的分离。

初始化

_subscribeCommand = [[RACCommand alloc] initWithEnabled:self.emailValidSignal 
signalBlock:^RACSignal *(id input) {
      return [SubscribeViewModel postEmail:email];
}];

signalBlock必须要返回一个信号,不能传nil,如果不想传递信号,则可以创建空的信号

 [RACSignal empty]

RAC

emailValidSignal用来指定创建的命令是否能够执行,self.emailValidSignal定义如下:

 _emailValidSignal = [RACObserve(self, email) map:^id(NSString *email) {
          return @([email isValidEmail]);
      }];

当self.emailValidSignal信号返回数据为yes的时候,_subscribeCommand才能够响应行为

signalBlock是当_subscribeCommand执行的时候会被调用的逻辑,该block返回的是一个包含了命令执行之后结果的signal。

属性

allowsConcurrentExecution代表当该命令正在执行的时候,是否能够再次触发改命令。

RACCommand内部是如何通过它去控制当前命令能不能够被执行以及能不能并发的执行:

以UIButton举例:

UIButton+RACCommandSupport.m

//UIButton的enabel和command的enabel绑定起来了,也就是如果command的enabel
//设置未false,那么button则不可用。
 disposable = [command.enabled setKeyPath:@keypath(self.enabled) onObject:self];

以下是RACCommand中有关allowsConcurrentExecution的源码:

if (enabledSignal == nil) {
       enabledSignal = [RACSignal return:@YES];
   } else {
       enabledSignal = [[[enabledSignal
           startWith:@YES]
           takeUntil:self.rac_willDeallocSignal]
           replayLast];
   }
   _immediateEnabled = [[RACSignal
       combineLatest:@[ enabledSignal, moreExecutionsAllowed ]]
       and];
 
   _enabled = [[[[[self.immediateEnabled
       take:1]
       concat:[[self.immediateEnabled skip:1] deliverOn:RACScheduler.
       mainThreadScheduler]]
       distinctUntilChanged]
       replayLast]
       setNameWithFormat:@"%@ -enabled", self]; 

整个command是否可用,是由immeditateEnable来决定的,而imediateEnabled是取enableSignal和moreExecutionsAllowed的与操作。

enableSignal是command初始化方法传进来的第一个参数。

RACSignal *immediateExecuting = [RACObserve(self, activeExecutionSignals) map:^
(NSArray *activeSignals) {
        return @(activeSignals.count > 0);
    }];
 
    RACSignal *moreExecutionsAllowed = [RACSignal
        if:RACObserve(self, allowsConcurrentExecution)
        then:[RACSignal return:@YES]
        else:[immediateExecuting not]];

imediateExecuting是当前正在执行未完成的命令数量的信号,如果command的allowsConcurrentExecution设置的yes,则moreExcutionsAllowed返回yes,反之则取决于有没有正在执行的信号,如果有就返回false,没有返回yes。一个信号是否执行完,根据signal有没有发出complete事件来判断的。

方法

excute

RACSignal *signal =[command execute:@2];

在command的初始化方法中,有一个signalBlock参数,该block有一个input参数,在command执行excute方法的时候传入的参数被当作signalBlock的input参数传到signalBlock中。比如通常将一个网络请求的封装成一个Command,此时网络请求的参数就可以通过调用excute的时候传入。

这种情况下,获取command执行之后返回的数据可以通过:

RACSignal *signal =[command execute:@2];
// 在这里就可以订阅信号了 
[signal subscribeNext:^(id x) { 
    NSLog(@"%@",x); 
}];

订阅命令完成的信号

[[self.viewModel.subscribeCommand execute:nil] subscribeCompleted:^{
  NSLog(@"The command executed");
}];

executionSignals

属于信号中的信号,在subscribeNext中返回的是信号x,再次订阅信号x才能获取具体数据

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

也可以直接使用excutionSignals.switchToLatest subscribeNext:获取到数据

     // 监听登录产生的数据
    [_LoginCommand.executionSignals.switchToLatest subscribeNext:^(id x) {
 
        if ([x isEqualToString:@"登录成功"]) {
            NSLog(@"登录成功");
        }
    }];
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容