Parquet是面向分析型业务的列式存储格式,由Twitter和Cloudera合做开发,2015年5月从Apache的孵化器里毕业成为Apache顶级项目,最新的版本是1.8.0。html
列式存储和行式存储相比有哪些优点呢?java
当时Twitter的日增数据量达到压缩以后的100TB+,存储在HDFS上,工程师会使用多种计算框架(例如MapReduce, Hive, Pig等)对这些数据作分析和挖掘;日志结构是复杂的嵌套数据类型,例如一个典型的日志的schema有87列,嵌套了7层。因此须要设计一种列式存储格式,既能支持关系型数据(简单数据类型),又能支持复杂的嵌套类型的数据,同时可以适配多种数据处理框架。git
关系型数据的列式存储,能够将每一列的值直接排列下来,不用引入其余的概念,也不会丢失数据。关系型数据的列式存储比较好理解,而嵌套类型数据的列存储则会遇到一些麻烦。如图1所示,咱们把嵌套数据类型的一行叫作一个记录(record),嵌套数据类型的特色是一个record中的column除了能够是Int, Long, String这样的原语(primitive)类型之外,还能够是List, Map, Set这样的复杂类型。在行式存储中一行的多列是连续的写在一块儿的,在列式存储中数据按列分开存储,例如能够只读取A.B.C这一列的数据而不去读A.E和A.B.D,那么如何根据读取出来的各个列的数据重构出一行记录呢?github
图1 行式存储和列式存储算法
Google的Dremel系统解决了这个问题,核心思想是使用“record shredding and assembly algorithm”来表示复杂的嵌套数据类型,同时辅以按列的高效压缩和编码技术,实现下降存储空间,提升IO效率,下降上层应用延迟。Parquet就是基于Dremel的数据模型和算法实现的。app
Parquet是语言无关的,并且不与任何一种数据处理框架绑定在一块儿,适配多种语言和组件,可以与Parquet配合的组件有:框架
查询引擎: Hive, Impala, Pig, Presto, Drill, Tajo, HAWQ, IBM Big SQL
计算框架: MapReduce, Spark, Cascading, Crunch, Scalding, Kite
数据模型: Avro, Thrift, Protocol Buffers, POJOs
那么Parquet是如何与这些组件协做的呢?这个能够经过图2来讲明。数据从内存到Parquet文件或者反过来的过程主要由如下三个部分组成:
1, 存储格式(storage format)
parquet-format项目定义了Parquet内部的数据类型、存储格式等。
2, 对象模型转换器(object model converters)
这部分功能由parquet-mr项目来实现,主要完成外部对象模型与Parquet内部数据类型的映射。
3, 对象模型(object models)
对象模型能够简单理解为内存中的数据表示,Avro, Thrift, Protocol Buffers, Hive SerDe, Pig Tuple, Spark SQL InternalRow等这些都是对象模型。Parquet也提供了一个example object model 帮助你们理解。
例如parquet-mr项目里的parquet-pig项目就是负责把内存中的Pig Tuple序列化并按列存储成Parquet格式,以及反过来把Parquet文件的数据反序列化成Pig Tuple。
这里须要注意的是Avro, Thrift, Protocol Buffers都有他们本身的存储格式,可是Parquet并无使用他们,而是使用了本身在parquet-format项目里定义的存储格式。因此若是你的应用使用了Avro等对象模型,这些数据序列化到磁盘仍是使用的parquet-mr定义的转换器把他们转换成Parquet本身的存储格式。
图2 Parquet项目的结构
理解Parquet首先要理解这个列存储格式的数据模型。咱们以一个下面这样的schema和数据为例来讲明这个问题。
message AddressBook { required string owner; repeated string ownerPhoneNumbers; repeated group contacts { required string name; optional string phoneNumber; } }
这个schema中每条记录表示一我的的AddressBook。有且只有一个owner,owner能够有0个或者多个ownerPhoneNumbers,owner能够有0个或者多个contacts。每一个contact有且只有一个name,这个contact的phoneNumber无关紧要。这个schema能够用图3的树结构来表示。
每一个schema的结构是这样的:根叫作message,message包含多个fields。每一个field包含三个属性:repetition, type, name。repetition能够是如下三种:required(出现1次),optional(出现0次或者1次),repeated(出现0次或者屡次)。type能够是一个group或者一个primitive类型。
Parquet格式的数据类型没有复杂的Map, List, Set等,而是使用repeated fields 和 groups来表示。例如List和Set能够被表示成一个repeated field,Map能够表示成一个包含有key-value 对的repeated field,并且key是required的。
图3 AddressBook的树结构表示
那么如何把内存中每一个AddressBook对象按照列式存储格式存储下来呢?
在Parquet格式的存储中,一个schema的树结构有几个叶子节点,实际的存储中就会有多少column。例如上面这个schema的数据存储实际上有四个column,如图4所示。
图4 AddressBook实际存储的列
Parquet文件在磁盘上的分布状况如图5所示。全部的数据被水平切分红Row group,一个Row group包含这个Row group对应的区间内的全部列的column chunk。一个column chunk负责存储某一列的数据,这些数据是这一列的Repetition levels, Definition levels和values(详见后文)。一个column chunk是由Page组成的,Page是压缩和编码的单元,对数据模型来讲是透明的。一个Parquet文件最后是Footer,存储了文件的元数据信息和统计信息。Row group是数据读写时候的缓存单元,因此推荐设置较大的Row group从而带来较大的并行度,固然也须要较大的内存空间做为代价。通常状况下推荐配置一个Row group大小1G,一个HDFS块大小1G,一个HDFS文件只含有一个块。
图5 Parquet文件格式在磁盘的分布
拿咱们的这个schema为例,在任何一个Row group内,会顺序存储四个column chunk。这四个column都是string类型。这个时候Parquet就须要把内存中的AddressBook对象映射到四个string类型的column中。若是读取磁盘上的4个column要可以恢复出AddressBook对象。这就用到了咱们前面提到的 “record shredding and assembly algorithm”。
对于嵌套数据类型,咱们除了存储数据的value以外还须要两个变量Repetition Level(R), Definition Level(D) 才能存储其完整的信息用于序列化和反序列化嵌套数据类型。Repetition Level和 Definition Level能够说是为了支持嵌套类型而设计的,可是它一样适用于简单数据类型。在Parquet中咱们只需定义和存储schema的叶子节点所在列的Repetition Level和Definition Level。
嵌套数据类型的特色是有些field能够是空的,也就是没有定义。若是一个field是定义的,那么它的全部的父节点都是被定义的。从根节点开始遍历,当某一个field的路径上的节点开始是空的时候咱们记录下当前的深度做为这个field的Definition Level。若是一个field的Definition Level等于这个field的最大Definition Level就说明这个field是有数据的。对于required类型的field必须是有定义的,因此这个Definition Level是不须要的。在关系型数据中,optional类型的field被编码成0表示空和1表示非空(或者反之)。
记录该field的值是在哪个深度上重复的。只有repeated类型的field须要Repetition Level,optional 和 required类型的不须要。Repetition Level = 0 表示开始一个新的record。在关系型数据中,repetion level老是0。
下面用AddressBook的例子来讲明Striping和assembly的过程。
对于每一个column的最大的Repetion Level和 Definition Level如图6所示。
图6 AddressBook的Max Definition Level和Max Repetition Level
下面这样两条record:
AddressBook { owner: "Julien Le Dem", ownerPhoneNumbers: "555 123 4567", ownerPhoneNumbers: "555 666 1337", contacts: { name: "Dmitriy Ryaboy", phoneNumber: "555 987 6543", }, contacts: { name: "Chris Aniszczyk" } } AddressBook { owner: "A. Nonymous" }
以contacts.phoneNumber这一列为例,"555 987 6543"这个contacts.phoneNumber的Definition Level是最大Definition Level=2。而若是一个contact没有phoneNumber,那么它的Definition Level就是1。若是连contact都没有,那么它的Definition Level就是0。
下面咱们拿掉其余三个column只看contacts.phoneNumber这个column,把上面的两条record简化成下面的样子:
AddressBook { contacts: { phoneNumber: "555 987 6543" } contacts: { } } AddressBook { }
这两条记录的序列化过程如图7所示:
图7 一条记录的序列化过程
若是咱们要把这个column写到磁盘上,磁盘上会写入这样的数据(图8):
图8 一条记录的磁盘存储
注意:NULL实际上不会被存储,若是一个column value的Definition Level小于该column最大Definition Level的话,那么就表示这是一个空值。
下面是从磁盘上读取数据并反序列化成AddressBook对象的过程:
1,读取第一个三元组R=0, D=2, Value=”555 987 6543”
R=0 表示是一个新的record,要根据schema建立一个新的nested record直到Definition Level=2。
D=2 说明Definition Level=Max Definition Level,那么这个Value就是contacts.phoneNumber这一列的值,赋值操做contacts.phoneNumber=”555 987 6543”。
2,读取第二个三元组 R=1, D=1
R=1 表示不是一个新的record,是上一个record中一个新的contacts。
D=1 表示contacts定义了,可是contacts的下一个级别也就是phoneNumber没有被定义,因此建立一个空的contacts。
3,读取第三个三元组 R=0, D=0
R=0 表示一个新的record,根据schema建立一个新的nested record直到Definition Level=0,也就是建立一个AddressBook根节点。
能够看出在Parquet列式存储中,对于一个schema的全部叶子节点会被当成column存储,并且叶子节点必定是primitive类型的数据。对于这样一个primitive类型的数据会衍生出三个sub columns (R, D, Value),也就是从逻辑上看除了数据自己之外会存储大量的Definition Level和Repetition Level。那么这些Definition Level和Repetition Level是否会带来额外的存储开销呢?实际上这部分额外的存储开销是能够忽略的。由于对于一个schema来讲level都是有上限的,并且非repeated类型的field不须要Repetition Level,required类型的field不须要Definition Level,也能够缩短这个上限。例如对于Twitter的7层嵌套的schema来讲,只须要3个bits就能够表示这两个Level了。
对于存储关系型的record,record中的元素都是非空的(NOT NULL in SQL)。Repetion Level和Definition Level都是0,因此这两个sub column就彻底不须要存储了。因此在存储非嵌套类型的时候,Parquet格式也是同样高效的。
上面演示了一个column的写入和重构,那么在不一样column之间是怎么跳转的呢,这里用到了有限状态机的知识,详细介绍能够参考Dremel。
列式存储给数据压缩也提供了更大的发挥空间,除了咱们常见的snappy, gzip等压缩方法之外,因为列式存储同一列的数据类型是一致的,因此可使用更多的压缩算法。
压缩算法 |
使用场景 |
Run Length Encoding |
重复数据 |
Delta Encoding |
有序数据集,例如timestamp,自动生成的ID,以及监控的各类metrics |
Dictionary Encoding |
小规模的数据集合,例如IP地址 |
Prefix Encoding |
Delta Encoding for strings |
Parquet列式存储带来的性能上的提升在业内已经获得了充分的承认,特别是当大家的表很是宽(column很是多)的时候,Parquet不管在资源利用率仍是性能上都优点明显。具体的性能指标详见参考文档。
Spark已经将Parquet设为默认的文件存储格式,Cloudera投入了不少工程师到Impala+Parquet相关开发中,Hive/Pig都原生支持Parquet。Parquet如今为Twitter至少节省了1/3的存储空间,同时节省了大量的表扫描和反序列化的时间。这两方面直接反应就是节约成本和提升性能。
若是说HDFS是大数据时代文件系统的事实标准的话,Parquet就是大数据时代存储格式的事实标准。
梁堰波,现就任于明略数据,开源爱好者,Apache Hadoop & Spark contributor。北京航空航天大学计算机硕士,曾就任于Yahoo!、美团网、法国电信,具有丰富的大数据、数据挖掘和机器学习领域的项目经验。
感谢丁晓昀对本文的审校。