咱们在《Apache Cassandra 简介》文章中介绍了 Cassandra 的数据模型相似于 Google 的 Bigtable,对应的开源实现为 Apache HBase,并且咱们在 《HBase基本知识介绍及典型案例分析》 文章中简单介绍了 Apache HBase 的数据模型。按照这个思路,Apache Cassandra 的数据模型应该和 Apache HBase 的数据模型很相似,那么这二者的数据存储模型是否是同样的呢?本文将为你们解答这些问题。咱们从 KeySpace -> Table -> Partition -> Row -> Cell 顺序介绍。本文基于 Apache Cassandra 3.11.4 源码进行介绍的,不一样版本可能有些不同。服务器
Cassandra 中的 KeySpace 概念和 RDBMS 里面的 DataBase 概念很相似,一个 KeySpace 包含多张表,通常将有关联的数据表放到同一个 KeySpace 下面。KeySpace 建立的时候能够指定副本策略,副本因子以及是否启用 CommitLog 机制(相似 HBase 中的 WAL)。spa
Cassandra 中表的概念和 RDBMS 很相似。不一样的是在 Cassandra 中属于同一张表的数据在物理上是分布在不一样节点上存储的,同一张表由多个 Partition 组成。code
Cassandra 通常是由多台节点组成的,每台节点负责必定范围的,若是使用 Murmur3hash 的时候,每一个节点负责的 Token 相似于下面那样:blog
因此 Token 范围为 -9223372036854775808 ~ -4611686018427387904 的数据存储在 A 节点;同理,Token 范围为 -4611686018427387903 ~ -1 之间的数据存储在 B节点,其余相似;每一个 Token 范围由多个 Partition 构成,每一个 Partition 由一行或多行数据组成,Partition 相似下面的:
其中,username 为 Partition key;type 为 Clustering key。那么在这种状况下,username = iteblog 的两条数据构成一个 Partition;另外两条构成分别构成两个 Partitions。在底层存储每一个 Partition 格式以下:
从上图能够看出,一个 Partition 是由 PartitionHeader、零个或多个 Row (格式在后面介绍)以及 EndPartition 三部分组成的。EndPartition 这个用于标记 Partition 结束,只占用一个字节,并使用 0x00000001 标记。PartitionHeader 的格式以下:排序
在底层存储中,多个 Partition 组成一个 SSTable(Sorted-String Table)文件。那么同一个 SSTable 文件中的数据数据是如何组织的呢?答案是按照 Partition Key 计算获得的 Token 升序排序的。索引
上面看出,Partition 里面包含了零个或多个 Row,这些 Row 对应的 Partition Key 是同样的。非 Static 的 Row 在磁盘存储的格式以下:
上面除了 flags 、Row Body Size 、 Previous Row Body Size 以及 Primary Key Liveness Timestamp 这四个字段必定会存在,其余字段须要知足条件才会存储。下面对上面字段进行介绍:get
若是存在的列没有占总列数的一半,则按照所有列的顺序保存存在的列在排序后列的索引位置;
若是存在的列占总列数超过一半,则按照所有列的顺序保存不存在的列在排序后列的索引位置。
可见,Cassandra 经过将列的信息(包括列的名称、类型、表名、keySpace等信息)保存到对应 SSTable 的 md-X-big-Statistics.db 文件中,相应的行只保存列是否存在的标记信息,这个能够节省存储空间的占用。注意,HBase 存储数据的时候每一个 Cell 都须要保存列名称和列族名称的。源码
非 Static Row 的底层存储格式已经在前面描述过,对于 Static Row 除了没有上图的 Clustering info 信息,其他都同样,因此这里就不介绍了。hash
上图中最后有 N 个 Cell,那多个 Cell 之间的顺序是如何保证的呢?答案是按照列的名称字典顺序升序排序的。好比咱们表的定义以下:it
CREATE TABLE iteblog ( user_id text, type text, action text, username text, age text, email text, PRIMARY KEY(user_id) );
那么 Cell 的顺序排列以下:
action -> age -> email -> type -> username
这个排序是经过 BTree 实现的,Row 的实现类为 BTreeRow。
Cell 就是每列数据的底层实现,Cell 里面包含了列的定义信息,好比是否被删除、是否过时、是否设置了时间戳等。在 Cassandra 里面,Column 有 Simple 和 Complex(CASSANDRA-8099引入的) 之分。non-frozen collection 或 UDT(用户自定义类型)的列是 ComplexColumn(Complex Cell)。
咱们正常使用的列就是属于这种类型的,它的底层存储格式以下:
若是列属于 non-frozen collection 或 UDT(用户自定义类型),那么这个属于 Complex Cell,它的底层存储格式以下:
能够看出,Complex Cell 和 Simple Cell 大部分很相似,下面只介绍不同的地方:
在 Cassandra 中, Complex Cell 的实现类是 ComplexColumnData。
原文连接 本文为云栖社区原创内容,未经容许不得转载。