3

UIView サブクラスに ( と の両方を試した) と プロパティがありUITextView logViewます。次のように、プロパティに KVO を追加しました。atomicnonatomicNSOperationQueue uploadQueueNSOperationQueueoperationsCount

[[self uploadQueue] addObserver:self
            forKeyPath:@"operationCount" 
               options:(NSKeyValueObservingOptionNew |
                        NSKeyValueObservingOptionOld)
               context:NULL];

オブザーバー関数は次のようになります。

- (void)observeValueForKeyPath:(NSString *)keyPath
                      ofObject:(id)object
                        change:(NSDictionary *)change
                       context:(void *)context
{
    if ([object isKindOfClass: [NSOperationQueue class]]) {
        NSLog(@"Keypath is: %@ change dictionary is: %@", keyPath, change);
        NSInteger testnew = [[change objectForKey: @"new"] integerValue];
        NSInteger testold = [[change objectForKey: @"old"] integerValue];
        if (testnew > testold) {
            [[self logView] setText: [NSString stringWithFormat: @"Uploading %d files", testnew]];
            objc_setAssociatedObject([self logView], @"max_value_of_uploads", [change objectForKey: @"new"], OBJC_ASSOCIATION_COPY);
        } else {
            NSInteger value = [objc_getAssociatedObject([self logView], @"max_value_of_uploads") integerValue]; 
            [[self logView] setText: [NSString stringWithFormat: @"Uploaded %d of %d files", testnew, value]];
        }
    }
}

uploadQueue塗りつぶしは次のように機能します。

   ...
   NSDictionary *iter;
        NSEnumerator *enumerator = [objects objectEnumerator];
        while (iter = [enumerator nextObject]) {
            Uploader *op = [[Uploader alloc] initWithFileName: [iter objectForKey: @"fileName"] 
                                                             FileID: [[iter objectForKey: @"fileID"] integerValue] 
                                                       AndSessionID: [self sess]];
            //Uploader *op = [[Uploader alloc] init];
            [[self uploadQueue] addOperation: op];
   ...

ブロックがなければif-else、すべて正常に機能します。キュー内の操作の数に関する NSLog メッセージを受け取りました。数値は問題ありません。しかし、そのif-elseブロックでは、次のようなクラッシュが発生します。

bool _WebTryThreadLock(bool), 0x10617fb0: Tried to obtain the web lock from a thread other than the main thread or the web thread. This may be a result of calling to UIKit from a secondary thread. Crashing now...
1   WebThreadLock
2   -[UITextView setText:]
3   -[SincViewController observeValueForKeyPath:ofObject:change:context:]
4   NSKeyValueNotifyObserver
5   NSKeyValueDidChange
6   -[NSObject(NSKeyValueObservingPrivate) _changeValueForKey:key:key:usingBlock:]
7   -[NSObject(NSKeyValueObservingPrivate) _changeValueForKey:usingBlock:]
8   ____NSOQDelayedFinishOperations_block_invoke_0
9   -[NSObject(NSKeyValueObservingPrivate) _changeValueForKey:key:key:usingBlock:]
10  -[NSObject(NSKeyValueObservingPrivate) _changeValueForKey:usingBlock:]
11  __NSOQDelayedFinishOperations
12  _dispatch_after_timer_callback
13  _dispatch_source_invoke
14  _dispatch_queue_invoke
15  _dispatch_worker_thread2
16  _pthread_wqthread
17  start_wqthread

ブロックでクラッシュしelseます。さらに、logView に変更が見られません。今後ともよろしくお願いいたします。

更新: performSelectorOnMainThreadテキストを設定すると、私は救われました。

4

2 に答える 2

4

UI を更新するときは、メイン スレッドにいることを確認してください。考えられる解決策は次のとおりです。

dispatch_async(dispatch_get_main_queue(), ^{
    if (testnew > testold) {
        [[self logView] setText: [NSString stringWithFormat: @"Uploading %d files", testnew]];
        objc_setAssociatedObject([self logView], @"max_value_of_uploads", [change objectForKey: @"new"], OBJC_ASSOCIATION_COPY);
    } else {
        NSInteger value = [objc_getAssociatedObject([self logView], @"max_value_of_uploads") integerValue]; 
        [[self logView] setText: [NSString stringWithFormat: @"Uploaded %d of %d files", testnew, value]];
    }
}); 

dispatch_asyncorの使用は、目的の実装に依存することに注意してdispatch_syncください。ただし、後者の場合、現在のスレッドを確認しない限り、デッドロックのリスクがあります。

参照: 誰かが UI コードを別のメソッドに移動することを提案するNSOperation、オブザーバー、およびスレッド エラー:

-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {
  [self performSelectorOnMainThread:@selector(dataLoadingFinished:) withObject:nil waitUntilDone:YES];
}

..これは素晴らしい代替ルートです。

于 2012-06-26T15:18:41.493 に答える
1

ここではデリゲート パターンを使用することをお勧めします。非常に高度な手法と緩いコードを混在させているようで、すぐに問題が発生する可能性があります。たとえば、連想オブジェクトは必要ないと思います。

ところで、バックグラウンド スレッドから UI にアクセスしているため、エラーが発生します。UI 呼び出し (setText:) を performSelectorOnMainThread:withObject:waitUntilDone: またはブロック内にラップする必要があります。

于 2012-06-26T15:09:12.393 に答える