博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
YYCache 源码学习(二):YYDiskCache
阅读量:6704 次
发布时间:2019-06-25

本文共 3489 字,大约阅读时间需要 11 分钟。

整体思路

从作者的《YYCache 设计思路》一文中可以看出,作者在设计YYDiskCache之前做了充分的测试:iPhone 6 64G 下,SQLite 写入性能比直接写文件要高,但读取性能取决于数据大小:当单条数据小于 20K 时,数据越小 SQLite 读取性能越高;单条数据大于 20K 时,直接写为文件速度会更快一些。

YYDiskCache的磁盘缓存结合使用了文件储存和数据库储存。

个人理解:在进行磁盘缓存的时候,会判断要储存数据的大小,如果数据小于20K,则直接存入数据库(数据储存到inline_data字段,此时filename为空)。如果数据大于20K,先把数据以文件形式进行存储,然后再在数据库中储存对应的文件名(此时inline_data为NULL,filename为文件地址),具体的可以结合下文中提到的磁盘缓存的文件结构来看。

磁盘缓存的核心类是YYKVStorage,他主要封装了文件储存操作和SQLite数据库的操作。YYDiskCache是对YYKVStorage的封装,抛出的API和内存缓存相似,都有数据读写和修剪内存。

磁盘缓存的文件结构
/* File: /path/      /manifest.sqlite      /manifest.sqlite-shm      /manifest.sqlite-wal      /data/           /e10adc3949ba59abbe56e057f20f883e           /e10adc3949ba59abbe56e057f20f883e      /trash/            /unused_file_or_folder  SQL: create table if not exists manifest (    key                 text,    filename            text,    size                integer,    inline_data         blob,    modification_time   integer,    last_access_time    integer,    extended_data       blob,    primary key(key) );  create index if not exists last_access_time_idx on manifest(last_access_time); */

这个结构我们不需要多说什么,只提一个小点,作者在path路径下面设计了一个/data/和一个/trash/。删除文件是一个比较耗时的操作,在删除文件的时候,先进行文件的移动,然后在一个子线程中处理要删掉的文件,提高了整体的效率。

实现 LRU

磁盘缓存对缓存淘汰算法的实现就比较简单了,因为每次存储都有对应的数据库记录,而且表中设计了last_access_time这个字段,我们可以直接使用数据库的排序语句就可以找到最不常用的文件了。

代码分析

1.

- (sqlite3_stmt *)_dbPrepareStmt:(NSString *)sql {    if (![self _dbCheck] || sql.length == 0 || !_dbStmtCache) return NULL;    sqlite3_stmt *stmt = (sqlite3_stmt *)CFDictionaryGetValue(_dbStmtCache, (__bridge const void *)(sql));    if (!stmt) {        int result = sqlite3_prepare_v2(_db, sql.UTF8String, -1, &stmt, NULL);        if (result != SQLITE_OK) {            if (_errorLogsEnabled) NSLog(@"%s line:%d sqlite stmt prepare error (%d): %s", __FUNCTION__, __LINE__, result, sqlite3_errmsg(_db));            return NULL;        }        CFDictionarySetValue(_dbStmtCache, (__bridge const void *)(sql), stmt);    } else {        sqlite3_reset(stmt);    }    return stmt;}

这个方法是提前生成了sql语句的句柄,可以理解成提前把sql语句编译成字节码留给后面的执行函数(当前不执行)。同时,作者使用

_dbStmtCache对语句进行缓存,下次使用时可以更快度的加载出来。

2.

- (BOOL)_dbClose {    if (!_db) return YES;        int  result = 0;    BOOL retry = NO;    BOOL stmtFinalized = NO;        if (_dbStmtCache) CFRelease(_dbStmtCache);    _dbStmtCache = NULL;        do {        retry = NO;        result = sqlite3_close(_db);        // 状态为busy或者lock        if (result == SQLITE_BUSY || result == SQLITE_LOCKED) {            if (!stmtFinalized) {                stmtFinalized = YES;                sqlite3_stmt *stmt;                //sqlite3_stmt *sqlite3_next_stmt(sqlite3 *pDb, sqlite3_stmt *pStmt);                //表示从数据库pDb中对应的pStmt语句开始一个个往下找出相应prepared语句,如果pStmt为nil,那么就从pDb的第一个prepared语句开始。                while ((stmt = sqlite3_next_stmt(_db, nil)) != 0) {                    //释放数据库中的prepared语句资源                    sqlite3_finalize(stmt);                    retry = YES;                }            }        } else if (result != SQLITE_OK) {            if (_errorLogsEnabled) {                NSLog(@"%s line:%d sqlite close failed (%d).", __FUNCTION__, __LINE__, result);            }        }    } while (retry);    _db = NULL;    return YES;}

这个是关闭数据库的方法,_dbStmtCache中缓存了我们使用的句柄,所以首先要释放掉了_dbStmtCache

在真正关闭数据库的代码中使用了do-while循环,因为一次访问数据库并不一定成功,数据库可能是busy或者lock的状态,所以要使用一个循环来多次访问。

如果为能关闭数据库,作者使用了sqlite3_next_stmt一个个的找出prepared语句,并使用sqlite3_finalize释放了prepared资源(防止内存泄露)。

其他的就没什么好说的了,主要就是一些sql语句的用法,这些大家看一下,碰到陌生的api谷歌一下就有了 ~ 具体的文件的操作,比较常用,看起来就容易很多。

转载地址:http://qbblo.baihongyu.com/

你可能感兴趣的文章
【深入剖析Tomcat笔记】第四篇 默认连接器
查看>>
ElasticSearch(1)-入门
查看>>
计算传播学在新闻和公共舆论领域的应用
查看>>
Go语言--基础语法笔记
查看>>
Android 中使用自定义字体的方法
查看>>
[原]RobotFrameWork(一)robotframework(python版)及Ride在ubuntu下安装
查看>>
2018-06-27随想
查看>>
Stream.findFirst的一个疑问
查看>>
深入理解java虚拟机(二)HotSpot Java对象创建,内存布局以及訪问方式
查看>>
PYTHON 模块
查看>>
软件开发模式对比(瀑布、迭代、螺旋、敏捷)
查看>>
css默认被后代inherite的属性列表
查看>>
酷客多郝宪玮:不够小程序化的企业,将错失最近5年的流量红利
查看>>
2017年淘客全新玩法——代理模式
查看>>
《开源安全运维平台OSSIM最佳实践》媒体推荐
查看>>
JavaScript服务器编程(对象属性枚举中应当避免原型污染问题)
查看>>
【ORACLE技术嘉年华PPT】MySQL压力测试经验
查看>>
用using取别名居然不支持泛型…
查看>>
NET也不能忽略基础
查看>>
ROR随想(2009年)
查看>>