lsm-tree 其主要思想源于 The Log-Structured Merge-Tree (LSM-Tree) 这篇 1996 年的考古 paper。其主要思想是通过将随机写入转换成顺序写入提高写入的性能。
本项目 ADLSM-Tree 是在学习了 leveldb 的原理和参考了部分 lsm-tree 的资料后自主实现一个简单的 lsm-tree 持久化键值存储引擎。
-
ADLSM-Tree 的 并发内存表
memtable
采用了std::shared_mutex
和std::map
的方式简单实现,没有采用 leveldb 中的无锁跳表std::atomic
和skiplist
。详见 doc/mem_table.md。 -
ADLSM-Tree 的 排序字符串表
sstable
结构基本与 leveldb 一致,但 bloom-filter 是整个 sstable 一个,而不是每个 block 一个,同时bloom-filter
采用的是标准的双哈希模拟多哈希的方式实现,没有采用 leveldb 是采用单哈希模拟多哈希的方式。详见 doc/sstable.md。 -
ADLSM-Tree 的 预写日志
wal
是通过向文件追加{k,v}
简单实现的,没有采用 leveldb 中的日志文件的方式。详见 doc/wal.md。 -
ADLSM-Tree 的 版本控制
revision
则有些复杂,每次minor-compaction
或者major-comacption
都会触发版本突变,首先每个磁盘上的sstable
都会在生成时计算整个文件的SHA-256
并作为其文件名,然后内存中会生成一个新的level
对象记录当前层级的所有sstable
的元数据信息,例如sstable
的min-key
和max-key
,以及sstable
的SHA-256
。然后将level
对象持久化到磁盘中,同样计算level
文件的SHA-256
作为level
文件名,然后将level
文件的SHA-256
和其层级记录到内存一个新的revision
对象中,并持久化revision
对象,同样计算revision
对象的SHA-256
,更新内存中current_rev
指向刚刚生成的revision
对象,完成版本突变。并没有采用 leveldb 中的manifest
文件去记录版本的变化。详见 doc/revision.md。 -
ADLSM-Tree 的 压实采用
Tiering
策略以减小写入放大,后续会尝试采用L-Leveling
等其他压实策略。详见 doc/compaction.md。 -
ADLSM-Tree 的缓存采用
LRU
策略。对block
数据块和sstable
分别进行缓存以提高读取性能,详见 doc/cache.md。
- gtest
- google-crc32c
- openssl
- fmt
- spdlog
安装依赖 (ArchLinux):
sudo pacman -S gtest
sudo pacman -S google-crc32c
sudo pacman -S openssl
sudo pacman -S fmt
sudo pacman -S spdlog
在45行评论一下
45行(同时修改原45行的文本内容)
mkdir build
cmake ..
make -j8