SDWebImage源码阅读

阅读别人的代码,主要目的当然是为了学习。如果直接下载源码,然后像教科书一样一个一个文件的阅读,常常会遇到这两个问题:1、作者为什么要用方案A,好像方案B也可以啊;2、我艹,这坨垃圾代码,换我1/10就搞定了。
罗马非一日建成,开源代码也不是一天就写成这个样子。忽略过程而只专注结果,往往会错过很多东西。于是,我开始换另一种阅读源码的方式:从提交日志追溯代码的更迭。


1

第一个版本的SDWebImage(那个时候还不是这个名字)只有4个文件,3个类组成。一个是缓存,包含内存缓存和磁盘缓存。内存缓存用的是NSMutableDictionary,磁盘则是文件,以url的md5作为文件名。清理机制很简单,收到内存警告时,就清掉内存缓存,磁盘缓存是7天。看文件最后修改时间,如果大于7天就删除,时机是程序退出前(监听UIApplicationWillTerminateNotification)。
第二个UIImageView的子类,设置一个URL。如果url在缓存中,就set image;如果没有,创建一个NSOperation下载。
第三个就是下载Operation。下载方式比较简陋,用的是-[NSData initWithContentsOfURL:]

2

作者加了一个Motivation提到,加载远程图片以前用的方法都是NSURLConnection的异步模式(sendAsynchronousRequest)。当时他用的是Three20,发现它的速度比Youtube慢了10倍。后来发现,原来NSURLConnection用的是epoll并且工作在主线程上,自然速度就跟不上。(epoll强调的是高并发,对单个连接而言,同步模式速度更快)

3

很贴心的把磁盘缓存的目录从NSDocumentDirectory挪到了NSCachesDirectory。

4

调用方式把子类方式改为Category,而把原来的子类改为加到UIImageView的第一个subview中。

5

可能觉得这种加到UIImageVIew第一个subview的hack方式不太好,而且有同一个url重复下载问题。于是乎搞了一个SDWebImageManager的单例。
然后,1.0就这么发布了。

6

将NSData下载图片放送改为NSURLConnection的同步模式,主要目的是让每次下载都强制更新,而不是用系统缓存。

7

作者有抛弃了NSOperation,改为NSURLConnect异步模式。原因嘛,NSOperation太重太慢(我觉得主要问题还是它对并行处理不好,如果前面一个连接被服务器stuck了,后面的就一直处于等待中)。作者找到了sendAsynchronousReqest慢点原因,connect默认塞到了NSEventTrackingRunLoopMode中,这是负责处理UI事件的runloop。改为塞到其它runloop就没这个问题了

    NSURLRequest *request = [[NSURLRequest alloc] initWithURL:url cachePolicy:NSURLRequestReloadIgnoringLocalCacheData timeoutInterval:15];
    self.connection = [[[NSURLConnection alloc] initWithRequest:request delegate:self startImmediately:NO] autorelease];
    // Ensure we aren't blocked by UI manipulations (default runloop mode for NSURLConnection is NSEventTrackingRunLoopMode)
    [connection scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:NSRunLoopCommonModes];
    [connection start];

附带一个好处是,下载到一半可以cancel

8

新增SDWebImageDecoder。后台IO线程先decode图片后,再通知主线程显示。decode不是什么特别的技术,如果图片很多,还是很有必要的。

9

内存cache之前,检查一下当前剩余内存。小于12M就清掉老的Cache。
不过后来用NSCache缓存内存图片,去掉了12M限制。

10

新加参数SDWebImageRefreshCached,对于头像之类的图片很适用。它先还是会用SDWebImage的cache查找,同时仍然发起一个HTTP请求,只不过这个NSURLRequest启用了NSURLRequestUseProtocolCachePolicy,由系统帮我们处理缓存,抓包后发现,request自动加上了If-Modified-Since,并返回了304错误。


SDWebImage的缓存策略从发布起到现在没有太大改变,依然是URL为key,内存+本地文件缓存,重点提高了对外扩展接口,多格式支持和图片解压。LUR提了好多年,换NSCache后,估计不会针对性的对其实现了。针对小文件,用数据库做缓存;目录图片太多时,分目录保存,这些我觉得都可以加上。

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

推荐阅读更多精彩内容