наблюдатьValueForKeyPath не вызывается

Я разрабатываю тестовое приложение, в котором у меня есть NSOperationQueue. Я создаю NSInvocationOperation и наблюдаю за свойством isFinished этой операции. Как ни странно, методObservValueForKeyPath вызывается только иногда. Я не могу понять изменения, которые я должен внести, чтобы вызывать его каждый раз. Пожалуйста помоги.

Вот код, который я написал:

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

person user2122350    schedule 12.08.2013    source источник
comment
Как вы проверяете, когда она вызывается и не вызывается, и является ли она одной и той же проверкой в ​​CreateOperationWithContext?   -  person Wain    schedule 12.08.2013
comment
Когда вы не видели уведомление, вы на 100% уверены, что операция была успешно создана, выполнена и завершена, даже если вы все еще не видели isFinished КВН? Например, если не удалось разрешить @selector(CreateOperationWithContext:), то операция никогда не будет создана. Или, если CreateOperationWithContext: не удалось завершить (бесконечный цикл и т. д.), операция также не будет завершена. Несмотря на это, я бы рекомендовал установить completionBlock для вашей операции, а не добавлять наблюдателя для ключа isFinished. Это чище и надежнее.   -  person Rob    schedule 12.08.2013
comment
Привет, Роб и Вейн, спасибо за ваши ответы. Функция CreateOperationWithContext: вызывается каждый раз и также успешно выполняется. Я обновляю базу данных в функции и каждый раз вижу обновление. Не уведомляется только наблюдатель.   -  person user2122350    schedule 13.08.2013


Ответы (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:...) для того же keyPath, для которого уведомляется. Это может испортить внутренние структуры данных наблюдателя KVO и вызвать недетерминированные сбои, которые могут свести вас с ума. См. ответ, который я написал здесь для подробности.

person ipmcc    schedule 13.08.2013
comment
работал, постараюсь построить на этом. CreateOperationWithcontext выполняет обновление базы данных. Постараюсь укоротить функцию и протестировать. Спасибо - person user2122350; 13.08.2013