executeSelector в подклассе NSOperation - PullRequest
       24

executeSelector в подклассе NSOperation

2 голосов
/ 02 сентября 2011

Я не мог найти ответ где-либо еще в сети, поэтому любая помощь будет признательна.

Я пытаюсь создать систему, с помощью которой я могу получить результаты задачи NSOperation, что, как я понимаю, не может быть выполнено конкретными подклассами, такими как NSInvocation.

У меня есть подкласс NSOperation ( TheEngine ), который является абстрактным по соглашению и должен быть расширен для реализации функции -main, чтобы включить тело кода для выполнения.

TheEngine содержит следующую функцию инициализации, задачей которой является просто отметить theSelector и theObject, к которым принадлежит селектор. Он также регистрирует наблюдателя KV для свойства isFinished:

-(id)initWithCallbackSelector:(SEL)theSelector inObject:(id)theObject

В моей функции observeValueForKeyPath:ofObject:change:context: я бы хотел вызвать функцию обратного вызова следующим образом:

NSLog(@"Some debug text to ensure this function is being called", nil);
[theObject performSelector:theSelector withObject:someData afterDelay:0];

Весь процесс выглядит так:

aViewController запускает расширение TheEngine - скажем TheTask , вызывая следующее и добавляя его в очередь операций.

TheTask* TT = [[TheTask alloc] initWithCallbackSelector:
    @selector(resultHandler:) inObject:theObject];

Кажется, все работает, как и ожидалось, без каких-либо ошибок или исключений. Но когда выполнение достигает observeValueForKeyPath:ofObject:change:context:, обратный вызов фактически не вызывается. Я новичок в Obj-C, поэтому я не совсем уверен, правильно ли я понимаю этот тип потоков.

Вот весь код:

-(id)initWithCallbackSelector:(SEL)theSelector inObject:(id)theObject{

    if([self init]){

        self.selectorsParentObject      =   theObject;
        self.selectorToCallWhenFinished =   theSelector;


        [self addObserver:self forKeyPath:@"isFinished" options:NSKeyValueObservingOptionNew context:NULL];

        return self;
    }

    return nil; 
}


-(void)observeValueForKeyPath:(NSString*)keyPath ofObject:(id)theObject change:(NSDictionary*)theChange context:(void*)theContext{

    if([keyPath isEqualToString:@"isFinished"]){

        NSLog(@"activity is finished with change: %@", theChange);

        NSLog(@"target object: %@", self.selectorsParentObject);
        NSLog(@"target selector: %@", NSStringFromSelector(self.selectorToCallWhenFinished));

        //[self performSelectorOnMainThread:self.selectorToCallWhenFinished withObject:self.resultData waitUntilDone:NO];
        [self.selectorsParentObject performSelector:@selector(selectorToCallWhenFinished) withObject:self.resultData afterDelay:0];
    }
}

Любая помощь приветствуется!

Ответы [ 2 ]

1 голос
/ 02 сентября 2011

Ваш NSOperation, скорее всего, работает в фоновом потоке. Если этот поток пропадает, или если этот поток не может прокачать цикл выполнения, ваш вызов performSelector:withObject:afterDelay: не сработает. Вы прокомментировали звонок на performSelectorOnMainThread:.... Это сработало?

Возможно, вы должны запустить это в главном потоке или запустить с performSelector:withObject: (без afterDelay:). performSelector:withObject: не требует цикла выполнения.

0 голосов
/ 02 сентября 2011

Как предложил Роб, код выполнялся в фоновом потоке, как и вызов observeValueForKeyPath:ofObject:change:context:

. Сначала я изменил код, чтобы селектор сработал в главном потоке с помощью [self performSelectorOnMainThread:@selector(runCallback) withObject:nil waitUntilDone:NO];.

Но в этом случае основным потоком оказывается TheTask , и было выдано исключение, поскольку TheTask не владеет theSelector.Чтобы исправить это, я создал дополнительную функцию -runCallback и запустил ее из

-(void)observeValueForKeyPath:(NSString*)keyPath ofObject:(id)theObject change:(NSDictionary*)theChange context:(void*)theContext{

    if([keyPath isEqualToString:@"isFinished"]){

        NSLog(@"activity is finished with change: %@", theChange);

        NSLog(@"target object: %@", self.selectorsParentObject);
        NSLog(@"target selector: %@", NSStringFromSelector(self.selectorToCallWhenFinished));



        [self performSelectorOnMainThread:@selector(runCallback) withObject:nil waitUntilDone:NO];

        //[self performSelectorOnMainThread:self.selectorToCallWhenFinished withObject:self.resultData waitUntilDone:NO];
        //[self.selectorsParentObject performSelector:@selector(selectorToCallWhenFinished) withObject:self.resultData afterDelay:0];
    }
}

и в -runCallback:

-(void)runCallback{

    [self.selectorsParentObject performSelector:self.selectorToCallWhenFinished withObject:self.resultData afterDelay:0];

}

Это называется theSelector в Задача с правильными данными.Спасибо за участие :)

...