網(wǎng)絡(luò)框架研究

YTKNetwork

YTKNetwork封裝了AFNetworking,使用了Commond(命令)的設(shè)計模式,把一個請求封裝成了request對象。

批量處理 YTKBatchRequest

批量處理指的是同時發(fā)起多個請求,當(dāng)所有請求完成后再進(jìn)行處理,比如同時下載三張圖片,等所有圖片下載完成后再把三張圖片合成一張圖片。這個功能也可以使用dispatch group實現(xiàn)。

YTKBatchRequest是用來處理批量請求的類,它持有一個數(shù)組來保存所有的request,在start后會遍歷把所有的reqeust發(fā)起請求。當(dāng)所有的request都請求成功時,認(rèn)為這次批量請求完成;有一個request失敗了,就停止所有的請求,認(rèn)為這個批量請求失敗了。

AFNetworking

以前版本的AF里有這樣的代碼:

+ (void)networkRequestThreadEntryPoint:(id)__unused object {
    @autoreleasepool {
        [[NSThread currentThread] setName:@"AFNetworking"];

        NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
        [runLoop addPort:[NSMachPort port] forMode:NSDefaultRunLoopMode];
        [runLoop run];
    }
}

+ (NSThread *)networkRequestThread {
    static NSThread *_networkRequestThread = nil;
    static dispatch_once_t oncePredicate;
    dispatch_once(&oncePredicate, ^{
        _networkRequestThread = [[NSThread alloc] initWithTarget:self selector:@selector(networkRequestThreadEntryPoint:) object:nil];
        [_networkRequestThread start];
    });
    
    return _networkRequestThread;
}

上面的代碼是創(chuàng)建了一個線程并且開啟了它的runloop。這個線程就是問題里提到的常駐線程。
這個線程的作用是什么呢?

- (void)start {
    [self.lock lock];
    if ([self isReady]) {
        self.state = AFOperationExecutingState;
        
        [self performSelector:@selector(operationDidStart) onThread:[[self class] networkRequestThread] withObject:nil waitUntilDone:NO modes:[self.runLoopModes allObjects]];
    }
    [self.lock unlock];
}

- (void)operationDidStart {
    [self.lock lock];
    if (![self isCancelled]) {
        self.connection = [[NSURLConnection alloc] initWithRequest:self.request delegate:self startImmediately:NO];
        
        NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
        for (NSString *runLoopMode in self.runLoopModes) {
            [self.connection scheduleInRunLoop:runLoop forMode:runLoopMode];
            [self.outputStream scheduleInRunLoop:runLoop forMode:runLoopMode];
        }
        
        [self.connection start];
    }
    [self.lock unlock];
    
    dispatch_async(dispatch_get_main_queue(), ^{
        [[NSNotificationCenter defaultCenter] postNotificationName:AFNetworkingOperationDidStartNotification object:self];
    });
}

從上面的代碼里可以就看出,AF里面把每一個網(wǎng)絡(luò)請求的發(fā)起和解析都放在了這個線程里執(zhí)行。正常來說,一個線程執(zhí)行完任務(wù)后就退出了。開啟runloop是為了防止線程退出。一方面避免每次請求都要創(chuàng)建新的線程;另一方面,因為connection的請求是異步的,如果不開啟runloop,線程執(zhí)行完代碼后不會等待網(wǎng)絡(luò)請求完的回調(diào)就退出了,這會導(dǎo)致網(wǎng)絡(luò)回調(diào)的代理方法不執(zhí)行。

總結(jié):AFN 的做法是把網(wǎng)絡(luò)請求的發(fā)起和解析都放在同一個子線程中進(jìn)行,但由于子線程默認(rèn)不開啟 runloop,它在運行完所有代碼后退出線程。而網(wǎng)絡(luò)請求是異步的,這會導(dǎo)致獲取到請求數(shù)據(jù)時,線程已經(jīng)退出,代理方法沒有機(jī)會執(zhí)行。因此,AFN 的做法是使用一個 runloop 來保證線程不死,能保證代理方法的執(zhí)行。同時,避免每次請求都創(chuàng)建新的線程。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容