深入分析Parquet列式存儲格式

jopen 9年前發布 | 25K 次閱讀 Parquet

 

Parquet是面向分析型業務的列式存儲格式,由推ter和Cloudera合作開發,2015年5月從Apache的孵化器里畢業成為Apache頂級項目,最新的版本是1.8.0。

列式存儲

列式存儲和行式存儲相比有哪些優勢呢?

  1. 可以跳過不符合條件的數據,只讀取需要的數據,降低IO數據量。
  2. 壓縮編碼可以降低磁盤存儲空間。由于同一列的數據類型是一樣的,可以使用更高效的壓縮編碼(例如Run Length Encoding和Delta Encoding)進一步節約存儲空間。
  3. 只讀取需要的列,支持向量運算,能夠獲取更好的掃描性能。

當時推ter的日增數據量達到壓縮之后的100TB+,存儲在HDFS上,工程師會使用多種計算框架(例如MapReduce, Hive, Pig等)對這些數據做分析和挖掘;日志結構是復雜的嵌套數據類型,例如一個典型的日志的schema有87列,嵌套了7層。所以需要設計一種列式存儲格式,既能支持關系型數據(簡單數據類型),又能支持復雜的嵌套類型的數據,同時能夠適配多種數據處理框架。

關系型數據的列式存儲,可以將每一列的值直接排列下來,不用引入其他的概念,也不會丟失數據。關系型數據的列式存儲比較好理解,而嵌套類型數據的列存儲則會遇到一些麻煩。如圖1所示,我們把嵌套數據類型的一行叫做一個記錄(record),嵌套數據類型的特點是一個record中的column除了可以是Int, Long, String這樣的原語(primitive)類型以外,還可以是List, Map, Set這樣的復雜類型。在行式存儲中一行的多列是連續的寫在一起的,在列式存儲中數據按列分開存儲,例如可以只讀取A.B.C這一列的數據而不去讀A.E 和A.B.D,那么如何根據讀取出來的各個列的數據重構出一行記錄呢?

圖1 行式存儲和列式存儲

Google的 Dremel 系統解決了這個問題,核心思想是使用“record shredding and assembly algorithm”來表示復雜的嵌套數據類型,同時輔以按列的高效壓縮和編碼技術,實現降低存儲空間,提高IO效率,降低上層應用延遲。Parquet 就是基于Dremel的數據模型和算法實現的。

Parquet適配多種計算框架

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數據模型

理解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的樹結構表示

Parquet文件的存儲格式

那么如何把內存中每個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”。

Striping/Assembly算法

對于嵌套數據類型,我們除了存儲數據的value之外還需要兩個變量Repetition Level(R), Definition Level(D) 才能存儲其完整的信息用于序列化和反序列化嵌套數據類型。Repetition Level和 Definition Level可以說是為了支持嵌套類型而設計的,但是它同樣適用于簡單數據類型。在Parquet中我們只需定義和存儲schema的葉子節點所在列的 Repetition Level和Definition Level。

Definition Level

嵌套數據類型的特點是有些field可以是空的,也就是沒有定義。如果一個field是定義的,那么它的所有的父節點都是被定義的。從根節點開始遍歷,當某一個field的路徑上的節點開始是空的時候我們記錄下當前的深度作為這個field的Definition Level。如果一個field的Definition Level等于這個field的最大Definition Level就說明這個field是有數據的。對于required類型的field必須是有定義的,所以這個Definition Level是不需要的。在關系型數據中,optional類型的field被編碼成0表示空和1表示非空(或者反之)。

Repetition Level

記錄該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,也可以縮短這個上限。例如對于推ter的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現在為推ter至少節省了1/3的存儲空間,同時節省了大量的表掃描和反序列化的時間。這兩方面直接反應就是節約成本和提高性能。

如果說HDFS是大數據時代文件系統的事實標準的話,Parquet就是大數據時代存儲格式的事實標準。

參考文檔

  1. http://parquet.apache.org/
  2. https://blog.推ter.com/2013/dremel-made-simple-with-parquet
  3. http://blog.cloudera.com/blog/2015/04/using-apache-parquet-at-appnexus/
  4. http://blog.cloudera.com/blog/2014/05/using-impala-at-scale-at-allstate/

作者簡介

梁堰波,現就職于明略數據,開源愛好者,Apache Hadoop & Spark contributor。北京航空航天大學計算機碩士,曾就職于Yahoo!、美團網、法國電信,具備豐富的大數據、數據挖掘和機器學習領域的項目經驗。

 本文由用戶 jopen 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!