没有收到NSNotification

时间:2015-05-15 16:30:38

标签: ios objective-c nsnotifications

我在收到NSNotification时遇到问题。我可以告诉我,我的网络呼叫在服务器端是有效的,服务器的响应是在我的应用程序的网络层接收的,但是当我的网络层类完成向NSNotification发送UIViewController的最后一步时{1}},从未收到该通知。结果,我的应用程序挂起,即使其他一切都按预期工作。

这是启动所有内容的UIViewController:

- (void) updateAffiliations
{
    [self.activityIndicator startAnimating];
    self.activityIndicator.hidden = NO;

    Updater *updater = [[Updater alloc] init];
    NSDictionary *clearTextParams = [updater makeDictionary];
    NSString *phpPath = @"updateRecord.php";
    NSDictionary *parameters = [[SharedCipher sharedCipher] encryptParameters:clearTextParams];

    _preferenceUpdateResponse = @"preferencesResponse";

    [[NSNotificationCenter defaultCenter] addObserver : self
                                             selector : @selector(preferenceUpdateResult:)
                                                 name : _preferenceUpdateResponse
                                               object : nil];

    [[SharedNetworkObject sharedNetworkObject] sendHttpPost : phpPath
                                                 parameters : parameters
                                           notificationName : _preferenceUpdateResponse];
    NSLog(@"sent");
}

- (void) preferenceUpdateResult: (NSNotification *) notification
{
    NSLog(@"response received from network layer");
    NSDictionary *dict = [notification userInfo];

    [self.activityIndicator stopAnimating];
    self.activityIndicator.hidden = YES;

    //...
}

我所指的网络层是一个名为SharedNetworkObject的单例类。我已经反复使用过这个类,这是发生这个问题的唯一例子:

- (void) sendHttpPost : (NSString *) phpPath
           parameters : (NSDictionary *) parameters
     notificationName : (NSString *) notificationName
{
    NSLog(@"sending http post to : %@", phpPath);
    NSLog(@"params: %@", parameters);

    [self POST : phpPath
    parameters : parameters
       success : ^(NSURLSessionDataTask *task, id responseObject)
             {
                 NSLog(@"response object: %@", responseObject);
                 NSLog(@"broadcasting notification to : %@", notificationName);
                 [[NSNotificationCenter defaultCenter] postNotificationName:notificationName object:nil userInfo:responseObject];
             }

       failure : ^(NSURLSessionDataTask *task, NSError *error)
             {
                 NSLog(@"Shared Network Object error: %@", error);
                 NSDictionary * userInfo = @{ @"result" : @"failure", @"error" : error };
                 [[NSNotificationCenter defaultCenter] postNotificationName : notificationName object:nil userInfo:userInfo];
             }];
}

以下是控制台输出的示例,用于显示进度(然后是缺乏进度)。没有错误发生,但是自从上次收到通知后应用程序挂起:

  

将http帖子发送到:updateRecord.php
  params:{
      ciphertext =“忽略这一点,大量加密的乱码”;       “device_iv”=“0Q7yxsTpzQEaDGPpJ96JrA ==”;

      paddLength = 11;
      “session_id”= 845903271;
}
  送了   响应对象:{
      message =“成功更新记录”;
      结果=成功;
}   
广播通知:preferencesResponse

此输出看起来很好,但它不完整。应该有最后一行说

  

从网络层收到的响应

但是这从未输出,因为从未收到过通知。我检查了导致通知永远不会返回的所有常见原因,我认为这是正确的。事实上,我在其他地方三次使用这种精确的模式,它们都完美地工作。

任何人都能看到我在这里做错了什么?这可能是一个线程问题吗?

非常感谢!

2 个答案:

答案 0 :(得分:3)

由于网络代码在后台线程上执行,但您希望在主线程上接收通知(以更新您的ui和活动指示符),几乎可以肯定是一个线程问题。

幸运的是,简单的解决方案......将它们包装在dispatch_async调用中:

dispatch_async(dispatch_get_main_queue(), ^{
    [[NSNotificationCenter defaultCenter] postNotificationName:notificationName object:nil userInfo:responseObject];
});

答案 1 :(得分:0)

我终于弄明白了。我有一个先前的通知观察员,在发送此通知时尚未删除。我很惊讶这造成了一个问题,因为另一个通知名称与这个完全不同,但是一旦我重新排序代码以在我注册新代码之前移除其他观察者,问题就会消失。生活和学习,我猜......

感谢您的帮助,@ cbiggin!