没有收到 NSNotification

Not receiving NSNotification

我无法接收 NSNotification。我可以说我的网络调用在服务器端是有效的,并且服务器的响应是在我的应用程序的网络层接收到的,但是当我的网络层 class 执行发送 [=12= 的最后一步时] 到 UIViewController,从未收到该通知。结果,我的应用程序挂起,即使其他一切都按预期工作。

这是启动一切的 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 的单例 class。我已经多次使用此 class,这是唯一发生此问题的实例:

- (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];
             }];
}

这是一个控制台输出示例,用于显示进度(然后是没有进度)。没有出现错误,但由于未收到上次通知,应用程序挂起:

sending http post to : updateRecord.php
params: {
ciphertext = "ignore this, lots of encrypted jibberish";
"device_iv" = "0Q7yxsTpzQEaDGPpJ96JrA==";
paddLength = 11;
"session_id" = 845903271;
}
sent
response object: {
message = "successfully updated record";
result = success;
}
broadcasting notification to : preferencesResponse

这个输出看起来不错,只是它不完整。最后一行应该是

response received from network layer

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

有人看到我在这里做错了什么吗?这可能是线程问题吗?

非常感谢!

几乎可以肯定是线程问题,因为网络代码在后台线程上执行,但您希望在主线程上接收通知(以更新 ui 和 activity 指标)。

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

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

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

感谢@cbiggin 的帮助!