浅析YYCache
一、前言
读优秀的源码,对自己的提升还是很快的,无论是考虑问题的角度,还是编码能力。
带着问题读源码的,学习效率更高,可以暂时先定几个小问题,带着问题,去思考为什么作者这样弄,是否有替换方案?
1). 缓存用的是什么样的数据结构,是否方便?
2).缓存策略是什么?
3).缓存池大小是否有考虑?超出了限定大小,是怎么样的更新策略?
4).是否线程安全?
二、阅读理解
首先,Cache文件结构如下:

YYCache是对外的使用接口,具体有三种缓存方式:内存缓存,硬盘缓存,数据库缓存;
内存缓存用的是双向链表的数据结构;node记录信息有:指向上一个和下一个node的指针,node本身的key,value和其大小,时间,具体定义如下:
@interface _YYLinkedMapNode : NSObject {
@package
__unsafe_unretained _YYLinkedMapNode *_prev; // retained by dic
__unsafe_unretained _YYLinkedMapNode *_next; // retained by dic
id _key;
id _value;
NSUInteger _cost;
NSTimeInterval _time;
}
@end
为什么要定义成为双向链表呢?双向链表比起单项链表的唯一好处就是可以定位到父节点,所以一定有大量的骚操作需要定位其父节点。
_key,_value都是常规需求,_cost从字面了解应该是占用内存大小,_time从字面了解应该是记录了一个时间,这个时间只能是操作时间,有什么用,在这儿看不出来,但是从缓存的常规套路可以猜测,_cost是为了控制缓存池大小,_time是为了从时间角度控制缓存清除工作,比如,超过5天的缓存统一认为失效。
LinkedMap定义如下:
/**
A linked map used by YYMemoryCache.
It's not thread-safe and does not validate the parameters. Typically, you should not use this class directly.
*/
@interface _YYLinkedMap : NSObject {
@package
CFMutableDictionaryRef _dic; // do not set object directly
NSUInteger _totalCost;
NSUInteger _totalCount;
_YYLinkedMapNode *_head; // MRU, do not change it directly
_YYLinkedMapNode *_tail; // LRU, do not change it directly
BOOL _releaseOnMainThread;
BOOL _releaseAsynchronously;
}
LinkedMap就是对双向链表的增删查改操作,没多少可说的。
YYMemberCache:内存缓存的class。
其代码从条数、时间、内存大小三个层面进行了缓存更新控制。
刚开局就一个循环调用整理函数:
这儿注意的一个点是,dispatch_after是插入到后面的queue里面的block,延时执行,没有办法取消,所以必须在block中判断self是否还存在。
剩下的就是常规操作,作者期望可以5s自查一次,分别从占用内存,缓存条数,过期时间三个维度清理掉不符合要求的缓存。
- (void)_trimRecursively {
__weak typeof(self) _self = self;
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(_autoTrimInterval * NSEC_PER_SEC)), dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, 0), ^{
__strong typeof(_self) self = _self;
if (!self) return;
[self _trimInBackground];
[self _trimRecursively];
});
}
- (void)_trimInBackground {
dispatch_async(_queue, ^{
[self _trimToCost:self->_costLimit];
[self _trimToCount:self->_countLimit];
[self _trimToAge:self->_ageLimit];
});
}
接下来就是获取缓存和缓存两个核心动作,方法定义如下:
- (id)objectForKey:(id)key - (void)setObject:(id)object forKey:(id)key withCost:(NSUInteger)cost
通读代码,缓存策略是:
每次读写操作都会把对应的缓存node更新到链表首部,并且更新其时间(按照缓存更新逻辑,删除的肯定是不经常使用的缓存,经常活跃的缓存,尽量保留下来,所以是尾部删除,首部插入,这儿也就解释了为什么需要双向链表结构,就是方便尾部删除和更新node到链表首部)。
代码中通过互斥锁pthread_mutex来保证同一时刻只有一个操作在运行,所以是线程安全的。
** 后记
缓存逻辑其实分析到这儿就基本上完了,剩下的就是理解具体代码的功能,有意思的是作者在内存缓存中用的是pthread_mutex来保证线程安全,而在硬盘缓存中用的是信号量dispatch_semaphore_t ,信号量设置为1,可以模拟互斥锁,其性能如何,怎么玩的,可以移步到这儿
作者思路还是比较缜密的,如果非要说一个可以优化的点,是不是可以硬盘缓存的时候,写文件的操作放在app将要到后台的时间节点去做,尽量减少IO操作。
浅析YYCache的更多相关文章
- SQL Server on Linux 理由浅析
SQL Server on Linux 理由浅析 今天的爆炸性新闻<SQL Server on Linux>基本上在各大科技媒体上刷屏了 大家看到这个新闻都觉得非常震精,而美股,今天微软开 ...
- 【深入浅出jQuery】源码浅析--整体架构
最近一直在研读 jQuery 源码,初看源码一头雾水毫无头绪,真正静下心来细看写的真是精妙,让你感叹代码之美. 其结构明晰,高内聚.低耦合,兼具优秀的性能与便利的扩展性,在浏览器的兼容性(功能缺陷.渐 ...
- 高性能IO模型浅析
高性能IO模型浅析 服务器端编程经常需要构造高性能的IO模型,常见的IO模型有四种: (1)同步阻塞IO(Blocking IO):即传统的IO模型. (2)同步非阻塞IO(Non-blocking ...
- netty5 HTTP协议栈浅析与实践
一.说在前面的话 前段时间,工作上需要做一个针对视频质量的统计分析系统,各端(PC端.移动端和 WEB端)将视频质量数据放在一个 HTTP 请求中上报到服务器,服务器对数据进行解析.分拣后从不同的 ...
- Jvm 内存浅析 及 GC个人学习总结
从诞生至今,20多年过去,Java至今仍是使用最为广泛的语言.这仰赖于Java提供的各种技术和特性,让开发人员能优雅的编写高效的程序.今天我们就来说说Java的一项基本但非常重要的技术内存管理 了解C ...
- 从源码浅析MVC的MvcRouteHandler、MvcHandler和MvcHttpHandler
熟悉WebForm开发的朋友一定都知道,Page类必须实现一个接口,就是IHttpHandler.HttpHandler是一个HTTP请求的真正处理中心,在HttpHandler容器中,ASP.NET ...
- 【深入浅出jQuery】源码浅析2--奇技淫巧
最近一直在研读 jQuery 源码,初看源码一头雾水毫无头绪,真正静下心来细看写的真是精妙,让你感叹代码之美. 其结构明晰,高内聚.低耦合,兼具优秀的性能与便利的扩展性,在浏览器的兼容性(功能缺陷.渐 ...
- 浅析匿名函数、lambda表达式、闭包(closure)区别与作用
浅析匿名函数.lambda表达式.闭包(closure)区别与作用 所有的主流编程语言都对函数式编程有支持,比如c++11.python和java中有lambda表达式.lua和JavaScript中 ...
- word-break|overflow-wrap|word-wrap——CSS英文断句浅析
---恢复内容开始--- word-break|overflow-wrap|word-wrap--CSS英文断句浅析 一 问题引入 今天在再次学习 overflow 属性的时候,查看效果时,看到如下结 ...
随机推荐
- 机器学习算法概述第五章——CART算法
特点: 是一个二叉树,元素可以重复利用,可以做回归也可以做分类,分类用最小二乘法,即误差平方和最小 切割方法: 对于可量化的x来说: 切割点通常为两个x的平均值 左右两部分分别取均值,再评判以哪个分割 ...
- 3.24 7.13 Python基础汇总
对象类型 类型名称 示例 简要说明 备注 数字 int,float,complex 1234,3.14,1.3e5,3+4j 数字大小没有限制 十六进制用0x前缀和0-9,a-f表示 字符串 str ...
- 写代码 Log 也要认真点么?
Log自然是需要的, 尤其是正式的产品; 但如果只是自己或内部用用的小工具, 也需要认真点吗? 实话说, 自己对 log 总是不上心, 总觉得调试好了, 能跑了, 足以. 所以, 被大妈怼了好几次 l ...
- 数据库并发处理 - 上的一把好"锁"
为什么要有锁? 我们都是知道,数据库中锁的设计是解决多用户同时访问共享资源时的并发问题.在访问共享资源时,锁定义了用户访问的规则.根据加锁的范围,MySQL 中的锁可大致分成全局锁,表级锁和行锁三类. ...
- 从0开发3D引擎(四):搭建测试环境
目录 上一篇博文 了解自动化测试 单元测试 集成测试 端对端测试 通过打印日志来调试 了解运行测试 断点调试 通过Spector.js测试WebGL 通过log调试Shader 移动端测试 了解性能测 ...
- OBS Studio 完全开源免费录屏软件
OBS是Open Broadcaster Software的简称,它是一款永久免费的直播软件.OBS直播软件相比XSplit,占用资源相对较少,对配置要求相对要低一点,相同点就是录制格式MP4,不用再 ...
- Time、Date拼接成TimeStamp
Time.Date拼接成TimeStamp 有关于Time类型.Date类型的数据这里不再赘述,本文旨在讲解如何将数据库中的Time.Date类型取出来并转换成TimeStamp类型,话不多说,先看代 ...
- eclipse中使用postgreSQL报错( Cannot load JDBC driver class )
需求: 使用Maven插件调用PostgreSQL数据库 环境: eclipse_4.5.0+JDK_1.7+Tomcat_7.0+Maven+postgresql-9.1-901.jdbc4.jar ...
- 【移动测试】你的测试用例中,是否包含App前后台切换
App前后台切换是我们平时常用的一个操作,比如:按手机的home键将应用置于后台.直接按手机电源键关闭屏幕或者通过最近打开的应用列表切换应用等,由此,我们可以得出结论:当app置于前台时,它的页面对我 ...
- js六种数据类型
六种数据类型: undefined . boolean .string .number .object .function 效果地址:https://scrimba.com/c/cEedDGTd 代 ...