Наблюдение ValueForKeyPath не вызывается
Я разрабатываю тестовое приложение, в котором у меня есть NSOperationQueue. Я создаю NSInvocationOperation и наблюдаю за свойством isFinished этой операции. Странно, что visibleValueForKeyPath вызывается только иногда. Я не в состоянии понять изменения, которые я должен сделать, чтобы каждый раз вызывать его. Пожалуйста помоги.
Вот код, который я написал:
-(BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
........//initialization
queue = [NSOperationQueue new];
operation=[NSInvocationOperation new];
operation = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(CreateOperationWithContext:) object:context];
[operation addObserver:self forKeyPath:@"isFinished" options:(NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld) context:NULL];
[queue addOperation:operation];
..... // launch the view controller
}
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
if ([keyPath isEqualToString:@"isFinished"]) {
NSLog(@"came in");
[operation removeObserver:self forKeyPath:@"isFinished"];
}
else
{
[super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
}
}
1 ответ
Следующий код работает для меня. Я начал с шаблона приложения для одного представления iOS. Вот что у меня было:
@implementation SOAppDelegate
{
NSOperationQueue* queue;
NSOperation* operation;
}
- (void)CreateOperationWithContext: (id)foo
{
NSLog(@"Op ran");
}
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
queue = [NSOperationQueue new];
// operation = [NSInvocationOperation new]; // Commented this out because it's redundant with the next line
operation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(CreateOperationWithContext:) object:[NSObject new]];
[operation addObserver:self forKeyPath:@"isFinished" options:(NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld) context:NULL];
[queue addOperation:operation];
return YES;
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
if ([keyPath isEqualToString:@"isFinished"])
{
NSLog(@"came in");
[operation removeObserver:self forKeyPath:@"isFinished"];
}
else
{
[super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
}
}
// ... rest of empty default app delegate methods here...
@end
В консоли я вижу:
2013-08-13 08:04:15.150 TestOpKVO[71373:20b] Op ran
2013-08-13 08:04:21.903 TestOpKVO[71373:20b] came in
Итак, кое-что о вашей реализации -CreateOperationWithContext:
вызывает проблемы. Тем не менее, я все еще вижу, как вызывается уведомление KVO, даже если я изменяю операцию, чтобы вызвать исключение.
Если бы я был вами, я бы начал с этого очень простого, рабочего примера, а затем продолжал бы один шаг за раз адаптировать его к вашему реальному коду, проверяя на каждом этапе, чтобы убедиться, что уведомление все еще работает.
Несколько советов: (которые, вероятно, не связаны с проблемой, с которой вы сталкиваетесь, но хорошие практики для использования KVO)
Во-первых, используйте контексты KVO со своими наблюдениями. Это безопаснее и более детерминировано. Смотрите ответ, который я написал здесь для деталей.
Во-вторых, не звоните -removeObserver:forKeyPath:
изнутри зов -observeValueForKeyPath:
(или же -addObserver:...
либо) для того же ключевого пути, для которого отправляется уведомление. Это может испортить структуры данных внутреннего наблюдателя KVO и может вызвать недетерминированные сбои, которые могут свести вас с ума. Смотрите ответ, который я написал здесь для деталей.