没有收到 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 的帮助!
我无法接收 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 的帮助!