我在收到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
此输出看起来很好,但它不完整。应该有最后一行说
从网络层收到的响应
但是这从未输出,因为从未收到过通知。我检查了导致通知永远不会返回的所有常见原因,我认为这是正确的。事实上,我在其他地方三次使用这种精确的模式,它们都完美地工作。
任何人都能看到我在这里做错了什么?这可能是一个线程问题吗?
非常感谢!
答案 0 :(得分:3)
由于网络代码在后台线程上执行,但您希望在主线程上接收通知(以更新您的ui和活动指示符),几乎可以肯定是一个线程问题。
幸运的是,简单的解决方案......将它们包装在dispatch_async调用中:
dispatch_async(dispatch_get_main_queue(), ^{
[[NSNotificationCenter defaultCenter] postNotificationName:notificationName object:nil userInfo:responseObject];
});
答案 1 :(得分:0)
我终于弄明白了。我有一个先前的通知观察员,在发送此通知时尚未删除。我很惊讶这造成了一个问题,因为另一个通知名称与这个完全不同,但是一旦我重新排序代码以在我注册新代码之前移除其他观察者,问题就会消失。生活和学习,我猜......
感谢您的帮助,@ cbiggin!