Memstore
Memstore 概述web
Memstore是HBase框架中很是重要的组成部分之一,是HBase可以实现高性能随机读写相当重要的一环。深刻理解Memstore的工做原理、运行机制以及相关配置,对hbase集群管理、性能调优都有着很是重要的帮助shell
HBase中,Region是集群节点上最小的数据服务单元,用户数据表由一个或多个Region组成。在Region中每一个ColumnFamily的数据组成一个Store。每一个Store由一个Memstore和多个HFile组成,以下图所示:数组
以前咱们提到,HBase是基于LSM-Tree模型的,全部的数据更新插入操做都首先写入Memstore中(同时会顺序写到日志HLog中),达到指定大小以后再将这些修改操做批量写入磁盘,生成一个新的HFile文件,这种设计能够极大地提高HBase的写入性能;另外,HBase为了方便按照RowKey进行检索,要求HFile中数据都按照RowKey进行排序,Memstore数据在flush为HFile以前会进行一次排序,将数据有序化;还有,根据局部性原理,新写入的数据会更大几率被读取,所以HBase在读取数据的时候首先检查请求的数据是否在Memstore,写缓存未命中的话再到读缓存中查找,读缓存还未命中才会到HFile文件中查找,最终返回merged的一个结果给用户。缓存
可见,Memstore不管是对HBase的写入性能仍是读取性能都相当重要。其中flush操做又是Memstore最核心的操做,接下来重点针对Memstore的flush操做进行深刻地解析:首先分析HBase在哪些场景下会触发flush,而后结合源代码分析整个flush的操做流程,最后再重点整理总结和flush相关的配置参数,这些参数对于性能调优、问题定位都很是重要。安全
Memstore Flush触发条件服务器
HBase会在以下几种状况下触发flush操做,须要注意的是MemStore的最小flush单元是HRegion而不是单个MemStore。可想而知,若是一个HRegion中Memstore过多,每次flush的开销必然会很大,所以咱们也建议在进行表设计的时候尽可能减小ColumnFamily的个数网络
Memstore Flush流程负载均衡
为了减小flush过程对读写的影响,HBase采用了相似于两阶段提交的方式,将整个flush过程分为三个阶段:框架
Memstore 中的数据是排序的,当MemStore累计到必定阈值时 就会建立一个新的MemStore,而且将老的MemStore 添加到flush队列 flush时会将内存中不在版本范围内的数据所有删掉,而后再持久化到造成一个storeFile文件 而后后续的数据会写入一个新的storefile文件
StoreFile
StoreFile简介svg
1) 一个Region由多个Store组成,一个Store对应一个ColumnFamily(列族)
Store包括位于内存中的MemStore和位于磁盘的StoreFile;写操做先写入MemStore,当Memstore中的数据达到某个阈值,HRegionserver会启动flashcache进程写入StoreFile,每次写入造成单独的一个StoreFile
2)当StoreFile文件的数量增加到必定阈值后,系统会进行合并(minor、major compaction),在合并过程当中会进行版本合并和删除工做(majar),造成更大的StoreFile
3)当一个Region全部StoreFile的大小和数量超过必定阈值后,会把当前的Region分割为两个,并由HMaster分配到相应的HRegionserver服务器,实现负载均衡
4)客户端检索数据,先在MemStore找,找不到再找StoreFile
StoreFile以HFile格式保存在HDFS上
1)首先HFile文件是不定长的,长度固定的只有其中的两块:Trailer和FileInfo。正如图中所示的,Trailer中有指针指向其余数据块的起始点
2)File Info中记录了文件的一些Meta信息,例如:AVG_KEY_LEN, AVG_VALUE_LEN, LAST_KEY, COMPARATOR, MAX_SEQ_ID_KEY等
3)Data Index和Meta Index块记录了每一个Data块和Meta块的起始点
4)Data Block是HBase I/O的基本单元,为了提升效率,HRegionServer中有基于LRU的Block Cache机制。每一个Data块的大小能够在建立一个Table的时候经过参数指定,大号的Block有利于顺序Scan,小号Block利于随机查询。 每一个Data块除了开头的Magic之外就是一个个KeyValue对拼接而成, Magic内容就是一些随机数字,目的是防止数据损坏
5)HFile里面的每一个KeyValue对就是一个简单的byte数组。可是这个byte数组里面包含了不少项,而且有固定的结构。咱们来看看里面的具体结构:
开始是两个固定长度的数值,分别表示Key的长度和Value的长度。紧接着是Key,开始是固定长度的数值,表示RowKey的长度,紧接着是 RowKey,而后是固定长度的数值,表示Family的长度,而后是Family,接着是Qualifier(限定符),而后是两个固定长度的数值,表示Time Stamp和Key Type(Put/Delete)。Value部分没有这么复杂的结构,就是纯粹的二进制数据了
解释1(较为清晰)
解释2(较为简洁明了)
当一个Store中的StoreFile达到必定的阈值后,就会 进行一次合并(major compact),将对同一个key的修改合并到一块儿 造成一个大的StoreFile,当StoreFile大小达到必定的阈值后,又会对 StoreFile进行Split,等分红两个StoreFile