从控件中删除ReactiveCocoa信号

Die*_*ros 4 objective-c ios reactive-cocoa

如果我将信号分配给控件的属性:

    RAC(self.loginButton.enabled) = [RACSignal
            combineLatest:@[
                    self.usernameTextField.rac_textSignal,
                    self.passwordTextField.rac_textSignal
            ] reduce:^(NSString* username, NSString* password) {
                return @(username.length > 0 && password.length > 0);
            }];
Run Code Online (Sandbox Code Playgroud)

但是后来想分配一个不同RACSignalenabled,我怎么能在这之前清除任何现有的呢?

如果我尝试再次设置它,我会得到如下例外情况:

2013-10-29 16:54:50.623 myApp[3688:c07] *** Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'Signal <RACSignal: 0x975e9e0> name: +combineLatest: (
"<RACSignal: 0x975d600> name: <UITextField: 0x10f2c420> -rac_textSignal",
"<RACSignal: 0x975de30> name: <UITextField: 0x10f306e0> -rac_textSignal"
) reduce: is already bound to key path "self.loginButton.enabled" on object <LoginViewController: 0x10f264e0>, adding signal <RACSignal: 0x9763500> name: +combineLatest: (
"<RACSignal: 0x97624f0> name: <UITextField: 0x10f2c420> -rac_textSignal",
"<RACSignal: 0x97629e0> name: <UITextField: 0x10f306e0> -rac_textSignal"
) reduce: is undefined behavior'
Run Code Online (Sandbox Code Playgroud)

Jus*_*ers 29

ReactiveCocoa的很大一部分理念是消除国家.状态是随着时间的推移可以就地改变的任何东西,并且由于以下几个原因它是有问题的:

  1. 你失去了过去的信息.一旦变量被更改,就像之前的值从未存在过一样.
  2. 变化可以来自任何地方.很难看到有状态的代码,并确切地知道什么时候会发生 - 那里的地方很差.
  3. 并发和异步使状态管理变得困难,因为您现在必须协调多个执行点之间的更改.当有多个参与者可能相互冲突时,很难实现决定论.

RAC不允许对同一属性进行多次绑定的原因是它使得排序不确定.如果我有两个信号绑定enabled,哪一个优先?我怎么知道哪一个发送了最新值?

RAC不允许重新绑定相同财产的原因是它是一个有状态的事情.在原地更改绑定是必要的,并且由于上述所有原因而不好.

相反,使用信号作为表达随时间变化的声明方式.改变财产的一切 - 无论是现在还是将来 - 应该用一个信号来表示.

根据您的示例,很难确切知道这些输入是什么,但是假设您想根据a的值使用不同的登录文本字段UISwitch:

// A signal that automatically updates with the latest value of
// `self.emailLoginSwitch.on`.
RACSignal *emailLoginEnabled = [[[self.emailLoginSwitch
    rac_signalForControlEvents:UIControlEventValueChanged]
    mapReplace:self.emailLoginSwitch]
    map:^(UISwitch *switch) {
        return @(switch.on);
    }];

// Whether the user has entered a valid username and password.
RACSignal *usernameAndPasswordValid = [RACSignal
    combineLatest:@[
        self.usernameTextField.rac_textSignal,
        self.passwordTextField.rac_textSignal
    ] reduce:^(NSString* username, NSString* password) {
        return @(username.length > 0 && password.length > 0);
    }];

// Whether the user has entered a valid email address.
RACSignal *emailValid = [self.emailTextField.rac_textSignal map:^(NSString *email) {
    return @(email.length > 0);
}];

// Uses different conditions for validity depending (ultimately) on the value of
// `self.emailLoginSwitch`.
RAC(self.loginButton, enabled) = [RACSignal
    if:emailLoginEnabled
    then:emailValid
    else:usernameAndPasswordValid];
Run Code Online (Sandbox Code Playgroud)

通过这种方式,无论输入实际是什么(用户名/密码或电子邮件),绑定仍然有效,并且我们避免了在运行时改变事物的任何需要.