2

私は次のAcceptCallBackメソッドを持っており、メソッドの実行中にUIActivityIndi​​catorを追加することを望んでいたため、[mvc performSelectorOnMainThread:@selector(invoke) withObject:nil waitUntilDone:YES];invokeUIを変更する方法です。そして[mvc performSelectorOnMainThread:@selector(hide) withObject:nil waitUntilDone:YES];、UIActivityIndi​​catorを削除するためのこの行があります。ただし、発生しているように見えるのは、invokeAcceptCallBackは実行が終了したときにのみ呼び出されるということです。AcceptCallBackinvokeは2つの異なるスレッドで実行されていないため、同時に実行できますか?

void AcceptCallBack(
                CFSocketRef socket,
                CFSocketCallBackType type,
                CFDataRef address,
                const void *data,
                void *info)
{
NSLog(@"Start Receiving...");

MasterViewController* mvc = (__bridge MasterViewController*)info;
[mvc performSelectorOnMainThread:@selector(invoke) withObject:nil waitUntilDone:YES];

CFReadStreamRef readStream = NULL;
CFWriteStreamRef writeStream = NULL;
CFIndex bytes;
UInt8 buffer[8192];
UInt8 * fileData;
UInt8 recv_len = 0;

/* The native socket, used for various operations */
CFSocketNativeHandle sock = *(CFSocketNativeHandle *) data;

/* Create the read and write streams for the socket */
CFStreamCreatePairWithSocket(kCFAllocatorDefault, sock,
                             &readStream, &writeStream);

if (!readStream || !writeStream) {
    close(sock);
    fprintf(stderr, "CFStreamCreatePairWithSocket() failed\n");
    return;
}

CFReadStreamOpen(readStream);
CFWriteStreamOpen(writeStream);


bool headerRead = false;
int dataWritten = 0;
NSMutableString* filename = NULL;


NSMutableString * header = [[NSMutableString alloc] init];

while (true) {
    memset(buffer, 0, sizeof(buffer));
    bytes = CFReadStreamRead(readStream, buffer, sizeof(buffer));

    recv_len += bytes;

    if (bytes < 0) {
        fprintf(stderr, "CFReadStreamRead() failed: %d\n", (int)bytes);
        close(sock);
        return;
    }
    if (bytes == 0) {
        break;
    }

    if (!headerRead) {
        for (int b=0; b<bytes; b++) {
            if (buffer[b] == '\n') {
                headerRead = true;
                NSLog(@"Header is: %@", header);

                NSArray *listItems = [header componentsSeparatedByString:@":"];
                filename = [[NSMutableString alloc] init];
                [filename appendString:[listItems objectAtIndex:2]];
                [filename replaceOccurrencesOfString:@"/" withString:@"" options:NSCaseInsensitiveSearch range:NSMakeRange(0, [filename length])];
                fileData = (UInt8*)malloc(sizeof(UInt8) * [[listItems objectAtIndex:3] intValue]);

                b++;
                memcpy(fileData, buffer + b, bytes - b);
                dataWritten = bytes - b;

                break;
            } else {
                [header appendFormat:@"%c", buffer[b]];
            }

        }
    } else {
        memcpy(fileData + dataWritten, buffer, bytes);
        dataWritten += bytes;
    }

}



NSString* docFile = [NSTemporaryDirectory() stringByAppendingPathComponent:filename];

NSData * outputData = [[NSData alloc] initWithBytes:fileData length:dataWritten];
if ([outputData writeToFile:docFile atomically:false] == YES) {
    NSLog(@"File received and successfully written out to file------------------------------");

    MasterViewController * thing = (__bridge MasterViewController*)info;

    [thing restClient:NULL loadedFile:docFile];

    NSString *destDir = @"/Slide2Me/";
    [[thing restClient] uploadFile:filename toPath:destDir
                     withParentRev:nil fromPath:docFile];
    [mvc performSelectorOnMainThread:@selector(hide) withObject:nil waitUntilDone:YES];




} else {
    NSLog(@"Failed to write received data to file");
}

}

編集 それで、私が望む結果を得るために私がやったことは、上記のすべてのコードを入れてdispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT,0)、メインスレッドに加えたい変更をそのように挟むことです....

void AcceptCallBack(
                    CFSocketRef socket,
                    CFSocketCallBackType type,
                    CFDataRef address,
                    const void *data,
                    void *info)
{


NSLog(@"Start Receiving...");

MasterViewController* mvc = (__bridge MasterViewController*)info;

[mvc performSelectorOnMainThread:@selector(invoke) withObject:nil waitUntilDone:YES];


dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT,0), ^{
      [the code above];

});

[mvc performSelectorOnMainThread:@selector(hide) withObject:nil waitUntilDone:YES];
}
4

1 に答える 1

6

GCDを使用してマルチスレッド化し、インジケーターの開始時と停止時にシグナルを送ることができます。PerformSelector ...は、関数が終了したときに実行されます。AcceptCallBackメソッドの呼び出し方法は示されていませんが、すでにメインスレッドにあると思います。その場合は、別のスレッドでAcceptCallbackを呼び出し、以下のコードを使用してメインスレッドで「invoke」メソッドを実行する必要があります。

dispatch_async(dispatch_get_main_queue(), ^{
    <do work here>
});

http://developer.apple.com/library/ios/#documentation/General/Conceptual/ConcurrencyProgrammingGuide/OperationQueues/OperationQueues.html#//apple_ref/doc/uid/TP40008091-CH102-SW1

編集:

私はこのようにします

static dispatch_queue_t processing_queue;
static dispatch_queue_t request_processing_queue() {
    if (processing_queue == NULL) {
        processing_queue = dispatch_queue_create("com.xxx.processing", 0);
    }
    return processing_queue;
}

void AcceptCallBack(
                    CFSocketRef socket,
                    CFSocketCallBackType type,
                    CFDataRef address,
                    const void *data,
                    void *info)
{

    __block MasterViewController* mvc = (__bridge MasterViewController*)info;

    dispatch_async(processing_queue(), ^{

       dispatch_async(dispatch_get_main_queue(), ^{
            [mvc invoke];
        });


       ..... < Do AcceptCallback Code Here >


       dispatch_async(dispatch_get_main_queue(), ^{
            [mvc hide];
        });

    });

}

警告:これは単なる擬似コードです。

于 2013-02-27T00:46:35.847 に答える