【问题标题】:What is a correct alternative for subscribing to signals in ReactiveCocoa for network calls?在 ReactiveCocoa 中订阅信号以进行网络调用的正确替代方法是什么?
【发布时间】:2014-07-04 14:50:38
【问题描述】:

我是 ReactiveCocoa 世界的新手,在阅读了 ReactiveCocoa here 的最佳实践之后,我知道我需要 "avoid explicit subscriptions and disposal" 但在所有关于网络和 ReactiveCocoa 的教程中,我看到了相同的模式:创建信号 (向服务器发出GETPOST 请求,解析结果,sendNextsendCompleted) -> subcsribeNext(对结果进行 UI 操作或其他操作)-> subscribeError。所以我们看到这里有一个明确的订阅,我认为这不好。

有没有一些更正确和概念上更纯粹的方法来做这个常见的事情? rac_liftSelector:withSignals: 或类似的东西?或者当我们处理网络调用和AFNetworking 时,我们应该始终使用这种标准订阅模式?详细的解释会很有帮助。

编辑:

在我的应用程序中,我主要获取调用,其中一些是依赖的,而另一些是单一的(绝大多数),例如 loginfetchWhateverpostWhatever。我用这样的相同模式构造的所有 API 调用(self - 是我的 API manager NVMAPI 类,它是 AFHTTPSessionManager 子类):

    -(RACSignal*)loginUserWithEmail:(NSString *)email andPassword:(NSString *)password
     {
        __block NSURLSessionDataTask* task;
        return [[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        task = [self GET:kUserLoginEndpoint
          parameters:@{@"email": email, @"password": password}
             success:^(NSURLSessionDataTask *task, id responseObject) {
                     NVMUser* user = [[NVMUser alloc] initWithDictionary:responseObject[@"user"]];
                   [subscriber sendNext:user];
                   [subscriber sendCompleted];
             } failure:^(NSURLSessionDataTask *task, NSError *error) {
                 [subscriber sendError:error];
             }];
    return [RACDisposable disposableWithBlock:^{
        [task cancel];
    }];
       }] replayLazily];
        }

我不使用MVVM,而是使用简单的 MVC。以下是我如何在视图控制器中进行 API 调用和创建信号:

[SVProgressHUD showWithStatus:@"Processing..." maskType:SVProgressHUDMaskTypeBlack];
[[[NVMAPI api] loginUserWithEmail:self.emailTextField.text
                               andPassword:self.passwordTextField.text]    
 subscribeNext:^(id x) {
                [self.activeUser setupWithUser:x];
                [SVProgressHUD dismiss];
                [self performSegueWithIdentifier:kLoginSeque sender:self];
 } 
     error:^(NSError *error) {
             [SVProgressHUD dismiss];
             [self showAlertWithText:error.localizedDescription title:@"Error"];
 }];

我使用相同方式的所有信号,例如:fetchCommentspostStatus 等。如果我有依赖调用,我使用 flattenMap。所以我很感兴趣 - 这是创建信号和使用它们的正确方法(简单subscribeNext)?或者这可以通过一些更正确和优雅的方式来实现?

编辑 2:

我看到的订阅的主要问题 - 我不知道如何使用它们为 tableview 实现分页。我有 fetchComment 加载分页 cmets 的方法 - 每页包含 15 个 cmets。我既不能使用subcribeNext 也不能使用RAC() 绑定,对吧?我该如何管理这种模式?

【问题讨论】:

  • 您是否有一些您特别希望看到重做的示例代码?很多这些东西很难抽象地谈论,所以了解现实世界的算法变化如何真正有益。
  • @JustinSpahr-Summers 感谢您的回复!请查看我的编辑。我编写代码示例,说明如何在我的应用程序中应用我的方法。
  • @JustinSpahr-Summers 我还添加了我遇到的主要问题。
  • 为什么说不能用-subscribeNext:来获取每页cmets呢?您可以在该订阅块中做任何您想做的事情,包括更新显示 cmets 的视图。

标签: ios objective-c ios7 afnetworking reactive-cocoa


【解决方案1】:

网络调用并不特殊,您可以像编写任何其他信号一样编写发出网络请求的信号。

【讨论】:

  • @HarryDeveloper1212 扩展 Erik 在这里所说的内容,适用于任何其他信号(如 rac_liftSelector:withSignals:)的任何内容都同样适用于发出网络请求的信号。您可以-catch: 它,将其包含在-flattenMap: 中,或者做任何其他事情将其合并到信号链中,如果您想避免订阅,最终以提升或RAC() 绑定结束。
猜你喜欢
  • 1970-01-01
  • 2010-11-12
  • 1970-01-01
  • 2016-06-26
  • 2021-01-13
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-03-01
相关资源
最近更新 更多