SDWebImage 源码学习笔记 ☞ SDWebImageDownloader

SDWebImage-源码学习笔记.png

前言

这是本系列的第 4 篇,本篇将主要介绍 SDWebImageDownloader 这个负责下载的类,当然还有一些相关类及协议,如: SDWebImageDownloadTokenSDWebImageDownloaderOperationSDWebImageDownloaderOperationInterface 等。

正文

开启正文描述之前,依旧先看 2 个重要的枚举:SDWebImageDownloaderOptionsSDWebImageDownloaderExecutionOrder,具体含义见下方代码注释。

// 控制下载过程的选项
typedef NS_OPTIONS(NSUInteger, SDWebImageDownloaderOptions) {
    // 降低下载任务在队列中的优先级
    SDWebImageDownloaderLowPriority = 1 << 0,
    // 图片将在下载过程中逐步展示,而不是等下载完成后才一次性展示
    SDWebImageDownloaderProgressiveDownload = 1 << 1,
    // 使用 NSURLCache,默认是不使用的
    SDWebImageDownloaderUseNSURLCache = 1 << 2,
    // 如果图片是从 NSURLCache 读取的,那么执行 completionHandler 的时候,回传 nil
    SDWebImageDownloaderIgnoreCachedResponse = 1 << 3,
    // 后台继续执行任务
    SDWebImageDownloaderContinueInBackground = 1 << 4,
    // 允许处理存储在 NSHTTPCookieStore 中的 Cookie
    SDWebImageDownloaderHandleCookies = 1 << 5,
    // 允许不受信任的 SSL 证书,生产环境慎用
    SDWebImageDownloaderAllowInvalidSSLCertificates = 1 << 6,
    // 提高下载任务在队列中的优先级
    SDWebImageDownloaderHighPriority = 1 << 7,
    // 缩放大图
    SDWebImageDownloaderScaleDownLargeImages = 1 << 8,
};

// 任务的执行顺序
typedef NS_ENUM(NSInteger, SDWebImageDownloaderExecutionOrder) {
    // 先进先出,也是队列的默认执行顺序
    SDWebImageDownloaderFIFOExecutionOrder,
    // 后进先出,栈的执行顺序
    SDWebImageDownloaderLIFOExecutionOrder
};

接下来就到了 SDWebImageDownloader 这个类,我不准备一个属性一个方法地按顺序讨论,而是先说创建方法,然后通过一个主要方法将主体串起来。先看创建方法吧!

对外其实只公开了一个创建单例的方法 sharedDownloader,仔细查看代码会发现,最终调用的是 - (nonnull instancetype)initWithSessionConfiguration: 这个初始化方法,主要做一些初始化工作,并创建一个新 session。如果使用单例方法创建 downloader,则只会有一个 session,而如果通过其他方法创建,则可能创建 session 之前已经有一个了,这时候就需要先将之前的 cancel 之后再创建新的。

+ (nonnull instancetype)sharedDownloader {
    static dispatch_once_t once;
    static id instance;
    dispatch_once(&once, ^{
        instance = [self new];
    });
    return instance;
}

- (nonnull instancetype)init {
    return [self initWithSessionConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]];
}

- (nonnull instancetype)initWithSessionConfiguration:(nullable NSURLSessionConfiguration *)sessionConfiguration {
    if ((self = [super init])) {
        // 执行下载任务的 operation
        _operationClass = [SDWebImageDownloaderOperation class];
        // 要求解压图片
        _shouldDecompressImages = YES;
        // 执行顺序,先进先出
        _executionOrder = SDWebImageDownloaderFIFOExecutionOrder;
        
        // 设置下载操作的队列,由于最大并发数是 6,所以此 queue 是 并发队列,如果是 1,则为串行队列。
        _downloadQueue = [NSOperationQueue new];
        _downloadQueue.maxConcurrentOperationCount = 6;
        _downloadQueue.name = @"com.hackemist.SDWebImageDownloader";
        
        _URLOperations = [NSMutableDictionary new];
        
        // 请求头的字段,可接受的文件类型
#ifdef SD_WEBP
        _HTTPHeaders = [@{@"Accept": @"image/webp,image/*;q=0.8"} mutableCopy];
#else
        _HTTPHeaders = [@{@"Accept": @"image/*;q=0.8"} mutableCopy];
#endif
        
        // 锁,这里使用了信号量
        _operationsLock = dispatch_semaphore_create(1);
        _headersLock = dispatch_semaphore_create(1);
        // 超时时间
        _downloadTimeout = 15.0;

        [self createNewSessionWithConfiguration:sessionConfiguration];
    }
    return self;
}

// 创建新的 session
- (void)createNewSessionWithConfiguration:(NSURLSessionConfiguration *)sessionConfiguration {
    // 为避免影响,先取消可能存在的下载任务
    [self cancelAllDownloads];

    // cancel 之前的 session,然后创建一个新的
    if (self.session) {
        [self.session invalidateAndCancel];
    }

    sessionConfiguration.timeoutIntervalForRequest = self.downloadTimeout;

    self.session = [NSURLSession sessionWithConfiguration:sessionConfiguration
                                                 delegate:self
                                            delegateQueue:nil];
}

然后,我们看一下主要方法 - (nullable SDWebImageDownloadToken *)downloadImageWithURL:url options:options progress:progressBlock completed:completedBlock。直接调用了添加进度与完成回调的方法,并将返回值作为结果返回。

添加进度与完成回调的方法我们稍后再议,先看一下调用时传入的 createCallback。就做了两件事:先创建一个 request,用于准备一些基础参数,然后,依据 request 创建 operation,详见代码注释 ☟。

- (nullable SDWebImageDownloadToken *)downloadImageWithURL:(nullable NSURL *)url
                                                   options:(SDWebImageDownloaderOptions)options
                                                  progress:(nullable SDWebImageDownloaderProgressBlock)progressBlock
                                                 completed:(nullable SDWebImageDownloaderCompletedBlock)completedBlock {
    __weak SDWebImageDownloader *wself = self;

    return [self addProgressCallback:progressBlock completedBlock:completedBlock forURL:url createCallback:^SDWebImageDownloaderOperation *{
        
        __strong __typeof (wself) sself = wself;
        NSTimeInterval timeoutInterval = sself.downloadTimeout;
        if (timeoutInterval == 0.0) {
            timeoutInterval = 15.0;
        }

// *** 1.创建 request
        
        // 为避免重复缓存 (NSURLCache + SDImageCache) ,如果没有明确要求使用 NSURLCache,我们默认忽略本地缓存
        NSURLRequestCachePolicy cachePolicy = options & SDWebImageDownloaderUseNSURLCache ? NSURLRequestUseProtocolCachePolicy : NSURLRequestReloadIgnoringLocalCacheData;
        NSMutableURLRequest *request = [[NSMutableURLRequest alloc] initWithURL:url
                                                                    cachePolicy:cachePolicy
                                                                timeoutInterval:timeoutInterval];
        // The default is YES - in other words, cookies are sent from and stored to the cookie manager by default.
        request.HTTPShouldHandleCookies = (options & SDWebImageDownloaderHandleCookies);
        request.HTTPShouldUsePipelining = YES;
        
        // 设置 header,headersFilter 是过滤头部参数的block
        if (sself.headersFilter) {
            request.allHTTPHeaderFields = sself.headersFilter(url, [sself allHTTPHeaderFields]);
        } else {
            request.allHTTPHeaderFields = [sself allHTTPHeaderFields];
        }
        
// *** 2.创建下载的 operation (这个 operationClass ,给他赋什么值,他就是什么,如果不设置,就是默认值:[SDWebImageDownloaderOperation class])
        
        SDWebImageDownloaderOperation *operation = [[sself.operationClass alloc] initWithRequest:request
                                                                                       inSession:sself.session
                                                                                         options:options];
        
        operation.shouldDecompressImages = sself.shouldDecompressImages;
        
        // NSURLCredential 身份认证
        if (sself.urlCredential) {
            operation.credential = sself.urlCredential;
        } else if (sself.username && sself.password) {
            // NSURLCredentialPersistenceForSession: Credential should be stored only for this session.
            operation.credential = [NSURLCredential credentialWithUser:sself.username password:sself.password persistence:NSURLCredentialPersistenceForSession];
        }
        
        // 设置优先级
        if (options & SDWebImageDownloaderHighPriority) {
            operation.queuePriority = NSOperationQueuePriorityHigh;
        } else if (options & SDWebImageDownloaderLowPriority) {
            operation.queuePriority = NSOperationQueuePriorityLow;
        }
        
        // 更改执行顺序:先进后出(可在此设置) or 先进先出(默认)
        if (sself.executionOrder == SDWebImageDownloaderLIFOExecutionOrder) {
            // 通过反向设置依赖,指定了队列中任务的执行顺序先加进去的依赖于后加进去的,那就成了后进先出了😎
            [sself.lastAddedOperation addDependency:operation];
            sself.lastAddedOperation = operation;
        }

        return operation;
    }];
}

现在,我们来看看添加进度与完成回调的方法:

- (nullable SDWebImageDownloadToken *)addProgressCallback:(SDWebImageDownloaderProgressBlock)progressBlock
                                           completedBlock:(SDWebImageDownloaderCompletedBlock)completedBlock
                                                   forURL:(nullable NSURL *)url
                                           createCallback:(SDWebImageDownloaderOperation *(^)(void))createCallback {
    // The URL will be used as the key to the callbacks dictionary so it cannot be nil. If it is nil immediately call the completed block with no image or data.
    if (url == nil) {
        if (completedBlock != nil) {
            completedBlock(nil, nil, nil, NO);
        }
        return nil;
    }
    
    LOCK(self.operationsLock);
    
    SDWebImageDownloaderOperation *operation = [self.URLOperations objectForKey:url];
    
    // 如果是第 1 次进来,通过 url 是取不出 URLOperation 的,但是第 2 次就有可能找到,也就是想要重复发第 2 次请求的话,就可以取到。
    // 第 2 次可以取到(并且已经完成的情况下),则不会走括号里边,也就不会执行关键步骤:[self.downloadQueue addOperation:operation]; ,所以就不会发起请求了,因为将 operation 添加到队列的时候,系统会自动触发请求。
    if (!operation || operation.isFinished) {
        // 创建 operation
        operation = createCallback();
        __weak typeof(self) wself = self;
        operation.completionBlock = ^{
            __strong typeof(wself) sself = wself;
            if (!sself) {
                return;
            }
            
            LOCK(sself.operationsLock);
            [sself.URLOperations removeObjectForKey:url];
            UNLOCK(sself.operationsLock);
        };
        [self.URLOperations setObject:operation forKey:url];
        
        // 添加到队列,即开始执行!
        // Add operation to operation queue only after all configuration done according to Apple's doc.
        // `addOperation:` does not synchronously execute the `operation.completionBlock` so this will not cause deadlock.
        [self.downloadQueue addOperation:operation];
    }
    
    UNLOCK(self.operationsLock);

    // 存放进度和完成回调的 数组 array
    id downloadOperationCancelToken = [operation addHandlersForProgress:progressBlock completed:completedBlock];
    
    // 与下载任务关联的一个对象,用于取消操作的时候
    SDWebImageDownloadToken *token = [SDWebImageDownloadToken new];
    token.downloadOperation = operation;
    token.url = url;
    token.downloadOperationCancelToken = downloadOperationCancelToken;

    return token;
}

主要做了这么几件事:

  • 开始依然是参数校验

  • 然后从 self.URLOperation 里边取 operation

  • 第一次进来当然取不到 operation,于是就会进入 if (!operation || operation.isFinished) { ... } 的代码块。先执行我们传入的 createCallback() 创建 operation,然后将 operation 加入到 self.URLOperations 里边,同时设置好 operation 的 completionBlock,到时将 operation 移除,最后将 operation 加入到操作队列里,就会自动开始执行了。

  • 创建一个 token,他是 SDWebImageDownloadToken 的实例,将它与 operation、url、progressBlock 及 completedBlock 关联起来,用于后边之后的取消操作。其中 progressBlock 及 completedBlock 是通过 downloadOperationCancelToken 与 token 关联起来的,这里用到了 operation 中的一个方法:

// SDWebImageDownloaderOperation

- (nullable id)addHandlersForProgress:(nullable SDWebImageDownloaderProgressBlock)progressBlock
                            completed:(nullable SDWebImageDownloaderCompletedBlock)completedBlock {
    
    SDCallbacksDictionary *callbacks = [NSMutableDictionary new];
    
    if (progressBlock) callbacks[kProgressCallbackKey] = [progressBlock copy];
    
    if (completedBlock) callbacks[kCompletedCallbackKey] = [completedBlock copy];
    
    LOCK(self.callbacksLock);
    
    [self.callbackBlocks addObject:callbacks];
    
    UNLOCK(self.callbacksLock);
    
    return callbacks;
}

由此可知,这个 id downloadOperationCancelToken 是一个存放 progressBlock 和 completedBlock 的 dictionary。

这里 if 语句起到了一个非常重要的作用,即 避免重复下载相同数据,具体原因就不解释了,上边的代码注释里已经写了。

到这里是不是觉得少了点什么,是的,SDWebImageDownloaderOperationSDWebImageDownloadToken 的具体实现还不知道呢,接下来我们就分别查看这 2 个类,先从简单的开始吧!

SDWebImageDownloadToken

这个类只有 3 个属性,前边都用到了,属性声明如下:

// 下载任务对应的 url
@property (nonatomic, strong, nullable) NSURL *url;
// 实际是包含 progressBlock 和 completionBlock 的字典,是通过 `addHandlersForProgress:completed:` 返回的,用于取消操作
@property (nonatomic, strong, nullable) id downloadOperationCancelToken;
// 操作的 operation,继承自 NSOperation,不过又遵守了 `SDWebImageDownloaderOperationInterface` 这个协议,扩展了一些方法。
@property (nonatomic, weak, nullable) NSOperation<SDWebImageDownloaderOperationInterface> *downloadOperation;

它只实现了一个协议方法 cancel(Protocol: SDWebImageOperation),其中 self.downloadOperationCancelToken就是存放 progressBlock 和 completionBlock 的字典,然后将这个 token 创递给了 operation 的 cancel: 方法(也是一个协议方法),这个方法的具体实现下边就会说到。

- (void)cancel {
    if (self.downloadOperation) {
        SDWebImageDownloadToken *cancelToken = self.downloadOperationCancelToken;
        if (cancelToken) {
            [self.downloadOperation cancel:cancelToken];
        }
    }
}
SDWebImageDownloaderOperationInterface

在开始介绍 operation 之前,先看看他遵守的协议 SDWebImageDownloaderOperationInterface,声明了以下协议方法。如果想要使用自定义的 operation,则它必须继承自 NSOperation 并且遵守这个协议。这些方法的实现可以参考 SDWebImageDownloaderOperation

// SDWebImageDownloaderOperationInterface
// 初始化方法
- (nonnull instancetype)initWithRequest:(nullable NSURLRequest *)request
                              inSession:(nullable NSURLSession *)session
                                options:(SDWebImageDownloaderOptions)options;
// 保存进度和完成的回调
- (nullable id)addHandlersForProgress:(nullable SDWebImageDownloaderProgressBlock)progressBlock
                            completed:(nullable SDWebImageDownloaderCompletedBlock)completedBlock;
// 是否需要解压图片
- (BOOL)shouldDecompressImages;
- (void)setShouldDecompressImages:(BOOL)value;
// 凭证或称证书信息
- (nullable NSURLCredential *)credential;
- (void)setCredential:(nullable NSURLCredential *)value;
// 取消
- (BOOL)cancel:(nullable id)token;
SDWebImageDownloaderOperation

现在开始讨论 SDWebImageDownloaderOperation 这个类,下边是初始化方法:

- (nonnull instancetype)init {
    return [self initWithRequest:nil inSession:nil options:0];
}
// 也是协议方法
- (nonnull instancetype)initWithRequest:(nullable NSURLRequest *)request
                              inSession:(nullable NSURLSession *)session
                                options:(SDWebImageDownloaderOptions)options {
    if ((self = [super init])) {
        _request = [request copy];
        _shouldDecompressImages = YES;
        _options = options;
        _callbackBlocks = [NSMutableArray new];
        _executing = NO;
        _finished = NO;
        _expectedSize = 0;
        _unownedSession = session;
        _callbacksLock = dispatch_semaphore_create(1);
        _coderQueue = dispatch_queue_create("com.hackemist.SDWebImageDownloaderOperationCoderQueue", DISPATCH_QUEUE_SERIAL);
    }
    return self;
}

下面介绍 2 个比较重要的方法:

  • 核心方法 start

这是重写父类 NSOperation 的 start 方法,添加了自定义的操作。这个方法不需要手动调用,在将 operation 添加到 operationQueue 中的时候,系统会自动调用其 start 方法。重写后的操作包括以下几点:

①检测操作是否已取消,如果取消了,重置数据后直接返回。

        if (self.isCancelled) {
            self.finished = YES;
            [self reset];
            return;
        }

②如果需要 App 进入后台时,继续执行下载操作,需要开启后台任务。并设置 ExpirationHandler,取消下载任务,并结束后台操作。

        // 如果需要App进入后台时,继续执行此操作,需要开启后台任务。
        Class UIApplicationClass = NSClassFromString(@"UIApplication");
        BOOL hasApplication = UIApplicationClass && [UIApplicationClass respondsToSelector:@selector(sharedApplication)];
        if (hasApplication && [self shouldContinueWhenAppEntersBackground]) {
            __weak __typeof__ (self) wself = self;
            UIApplication * app = [UIApplicationClass performSelector:@selector(sharedApplication)];
            self.backgroundTaskId = [app beginBackgroundTaskWithExpirationHandler:^{
                __strong __typeof (wself) sself = wself;

                if (sself) {
                    [sself cancel];

                    [app endBackgroundTask:sself.backgroundTaskId];
                    sself.backgroundTaskId = UIBackgroundTaskInvalid;
                }
            }];
        }

③获取或创建 session,然后创建 dataTask,

④若 dataTask 创建成功,启动下载任务 [self.dataTask resume];,然后执行一次 progressBlock,并发送开始下载的通知。

④若 dataTask 创建失败,直接调用完成回调,构建 error 信息并返回,然后重置数据。

⑤关闭可能存在的后台下载任务。

具体下载过程中的操作,都在 session 相关的那些协议方法里边,详见代码注释,这里就不啰嗦了。

  • 取消操作

最后看一下取消操作,即协议方法 - (BOOL)cancel:(nullable id)token; 的实现,将取消过程中用到的所有方法都展开就是下边这样:

- (BOOL)cancel:(nullable id)token {
    BOOL shouldCancel = NO;
    
    LOCK(self.callbacksLock);
    
    // 移除 token,即移除一个存储着 completionBlock 和 progressBlock 的字典
    [self.callbackBlocks removeObjectIdenticalTo:token];
    
    // 如果已经没有回调,就去执行整体的 cancel 操作
    if (self.callbackBlocks.count == 0) {
        shouldCancel = YES;
    }
    
    UNLOCK(self.callbacksLock);
    
    if (shouldCancel) {
        // *** 点开 ☟
        [self cancel];
    }
    
    return shouldCancel;
}

- (void)cancel {
    @synchronized (self) {
        // *** 点开 ☟
        [self cancelInternal];
    }
}

- (void)cancelInternal {
    if (self.isFinished) return;
    [super cancel];

    if (self.dataTask) {
        // 取消下载任务,并发出停止的通知
        [self.dataTask cancel];
        __weak typeof(self) weakSelf = self;
        dispatch_async(dispatch_get_main_queue(), ^{
            [[NSNotificationCenter defaultCenter] postNotificationName:SDWebImageDownloadStopNotification object:weakSelf];
        });

        // As we cancelled the task, its callback won't be called and thus won't
        // maintain the isFinished and isExecuting flags.
        if (self.isExecuting) self.executing = NO;
        if (!self.isFinished) self.finished = YES;
    }
    // *** 点开 ☟
    [self reset];
}

// 重置变量
- (void)reset {
    LOCK(self.callbacksLock);
    [self.callbackBlocks removeAllObjects];
    UNLOCK(self.callbacksLock);
    self.dataTask = nil;
    
    if (self.ownedSession) {
        [self.ownedSession invalidateAndCancel];
        self.ownedSession = nil;
    }
}

小结

SDWebImageDownloader 的内容就先介绍到这类,其他细节见 HHSDWebImageStudy 中的源码注释。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352