王浩,葛昂,趙晴
(華北計(jì)算機(jī)系統(tǒng)工程研究所,北京 100083)
摘要:為了實(shí)現(xiàn)將傳統(tǒng)關(guān)系型數(shù)據(jù)庫(kù)中的增量數(shù)據(jù)快速導(dǎo)入同構(gòu)或者異構(gòu)目的庫(kù),在使用已有的增量提取方法的基礎(chǔ)上,提出了通過(guò)增加并行度和流式計(jì)算的方法加快同步速度。此方法不僅支持插入、更新和刪除的增量數(shù)據(jù)同步,而且可以抽取出數(shù)據(jù)庫(kù)表結(jié)構(gòu)信息動(dòng)態(tài)支持表結(jié)構(gòu)變更。與傳統(tǒng)單點(diǎn)抽取方式相比,大大提高了目的庫(kù)數(shù)據(jù)的新鮮度。
關(guān)鍵詞:增量同步; Spark; 流式計(jì)算
0引言
隨著大數(shù)據(jù)技術(shù)的發(fā)展,越來(lái)越多的企業(yè)開始構(gòu)建大數(shù)據(jù)平臺(tái)進(jìn)行數(shù)據(jù)處理。然而如何將保存在關(guān)系型數(shù)據(jù)庫(kù)中的數(shù)據(jù)快速同步到大數(shù)據(jù)平臺(tái)組件(例如HBase、HDFS)中,正成為很多企業(yè)面臨的問(wèn)題。Sqoop是常用的數(shù)據(jù)同步工具,其實(shí)質(zhì)是MapReduce任務(wù),延時(shí)較高,而且需要通過(guò)定時(shí)任務(wù)來(lái)達(dá)到自動(dòng)化流程效果。本文在觸發(fā)器記錄數(shù)據(jù)變化的基礎(chǔ)上,提出了一種使用Spark Streaming將增量數(shù)據(jù)抽取出來(lái),然后根據(jù)需要寫入到不同的目的庫(kù)的方法。由于只提取增量數(shù)據(jù),所以較Sqoop減少了數(shù)據(jù)量。另外由于是流式處理方式,降低了延時(shí)。
1增量提取
1.1增量提取的概念
增量提取是針對(duì)上一次提取而言,將上一次提取時(shí)間點(diǎn)到現(xiàn)在數(shù)據(jù)庫(kù)中插入、更新、刪除的數(shù)據(jù)提取出來(lái)[1]。
1.2常用的增量提取方法
1.2.1基于業(yè)務(wù)系統(tǒng)日志
在業(yè)務(wù)中將數(shù)據(jù)庫(kù)DML(Data Manipulation Language)語(yǔ)句輸出以日志的方式存儲(chǔ),然后通過(guò)解析日志將DML語(yǔ)句在目的庫(kù)中重放以達(dá)到目的。此方法需要侵入業(yè)務(wù)系統(tǒng),對(duì)于已經(jīng)成型的業(yè)務(wù)系統(tǒng)不適用。
1.2.2基于數(shù)據(jù)庫(kù)日志
解析數(shù)據(jù)庫(kù)日志也能達(dá)到增量提取的目的,但是各大數(shù)據(jù)庫(kù)廠商不對(duì)外開放數(shù)據(jù)庫(kù)系統(tǒng)的日志格式,這就使得解析日志變成了問(wèn)題。而且各數(shù)據(jù)庫(kù)的日志格式還不盡相同,難以達(dá)到通用性。
1.2.3基于觸發(fā)器
基于觸發(fā)器的方式,目前被廣泛運(yùn)用于數(shù)據(jù)庫(kù)增量提取。它通過(guò)在源表上建立插入、更新、刪除觸發(fā)器來(lái)記錄對(duì)數(shù)據(jù)的操作。每當(dāng)有數(shù)據(jù)變化時(shí),就會(huì)觸發(fā)相應(yīng)的觸發(fā)器,然后運(yùn)行觸發(fā)器定義的邏輯,將變化記錄到增量表。
1.3基于觸發(fā)器方法的具體實(shí)現(xiàn)
由于觸發(fā)器方法具有實(shí)現(xiàn)邏輯簡(jiǎn)單,對(duì)業(yè)務(wù)無(wú)入侵,數(shù)據(jù)庫(kù)通用等優(yōu)點(diǎn),所以本文采用了基于觸發(fā)器方式的增量提取方法。具體實(shí)現(xiàn)方法如下:
(1)創(chuàng)建名為dml_log的數(shù)據(jù)庫(kù)表,字段為id、table_name、record_id、execute_date、dml_type。其中id為自增id,table_name存儲(chǔ)要同步的源表表名稱,record_id是源表中發(fā)生變化的記錄的唯一標(biāo)識(shí),execute_date為觸發(fā)器執(zhí)行時(shí)的時(shí)間戳,dml_type為I、U、D分別代表insert、update、delete操作。
?。?)在源表上創(chuàng)建插入、更新、刪除類型的觸發(fā)器。創(chuàng)建語(yǔ)句在此省略。
2構(gòu)建Spark Streaming程序
2.1Spark Streaming
Spark是目前大數(shù)據(jù)處理領(lǐng)域比較常用的計(jì)算框架。它將中間計(jì)算結(jié)果維護(hù)在內(nèi)存中,這樣不僅可以做到中間結(jié)果的重用,而且減少了磁盤IO,大大加快了計(jì)算速度。Spark Streaming是構(gòu)建于Spark core之上的流式處理模塊。其原理是將流式數(shù)據(jù)切分成一個(gè)個(gè)小的片段,以mini batch的形式來(lái)處理這一小部分?jǐn)?shù)據(jù),從而模擬流式計(jì)算達(dá)到準(zhǔn)實(shí)時(shí)的效果。
2.2JdbcRDD
彈性分布式數(shù)據(jù)集(Resilient Distributed Datasets,RDD),它是Spark數(shù)據(jù)抽象的基石。RDD是一個(gè)只讀的分區(qū)記錄集合,分區(qū)分散在各個(gè)計(jì)算節(jié)點(diǎn)[2]。RDD提供了transformation和action兩類操作,其中transformation是lazy級(jí)別的,主要對(duì)數(shù)據(jù)處理流程進(jìn)行標(biāo)記,而不立即進(jìn)行運(yùn)算。action操作會(huì)觸發(fā)作業(yè)的提交,然后進(jìn)行回溯導(dǎo)致transformation操作進(jìn)行運(yùn)算。
JdbcRDD擴(kuò)展自RDD,是RDD的子類。內(nèi)部通過(guò)JDBC(Java Data Base Connectivity)操作以數(shù)據(jù)庫(kù)為源頭構(gòu)建RDD。其構(gòu)造函數(shù)簽名為:
class JdbcRDD[T: ClassTag](
sc: SparkContext,
getConnection:()=> Connection,
sql: String,
lowerBound: Long,
upperBound: Long,
numPartitions: Int,
mapRow:(ResultSet) => T =
JdbcRDD.resultSetToObjectArray _)
extends RDD[T](sc, Nil) with Logging {…}
2.3具體實(shí)現(xiàn)
Spark官方提供用于構(gòu)建Spark Streaming的數(shù)據(jù)源沒(méi)有對(duì)數(shù)據(jù)庫(kù)進(jìn)行支持,所以本文自己實(shí)現(xiàn)對(duì)數(shù)據(jù)庫(kù)的支持。編寫繼承自InputDStream類的DirectJdbcInputDStream類,其簽名為:
class DirectJdbcInputDStream[T: ClassTag](
@transient ssc_ : StreamingContext,
param: JdbcParam) extends
InputDStream[Row] (ssc_) with Logging {…}
對(duì)start()、compute()和stop()方法進(jìn)行重寫。
(1)在start函數(shù)中注冊(cè)JDBC驅(qū)動(dòng),用于JDBC獲取初始化信息(構(gòu)造JdbcRDD時(shí)的參數(shù));
(2)compute函數(shù)會(huì)被框架間隔指定的時(shí)間反復(fù)調(diào)用,其實(shí)質(zhì)是如何返回一個(gè)JdbcRDD。首先通過(guò)JDBC獲取本次需要拉取的trigger記錄的id的上下界以及表的Schema信息;然后以這些信息為參數(shù)生成提取真實(shí)數(shù)據(jù)的SQL,其邏輯為用選中的trigger表中的記錄和原表在record_id上進(jìn)行左連接;最后使用該SQL當(dāng)做參數(shù)構(gòu)建JdbcRDD。值得說(shuō)明的是,構(gòu)建JdbcRDD時(shí)是可以指定并行度的,每個(gè)worker節(jié)點(diǎn)都會(huì)建立到數(shù)據(jù)庫(kù)的JDBC連接,由多個(gè)節(jié)點(diǎn)并行去數(shù)據(jù)庫(kù)拉取屬于自己的那一部分?jǐn)?shù)據(jù),這就大大增加了提取和處理速度。
(3)在stop函數(shù)中關(guān)閉JDBC連接??傮w來(lái)看,就是在driver程序中執(zhí)行的JDBC程序獲取初始化參數(shù),在executor中執(zhí)行的JDBC程序拉取真實(shí)的數(shù)據(jù)。
(4)編寫driver程序:
val sc = new SparkContext(new SparkConf)
val ssc = new StreamingContext(sc, Seconds(30))
val directStream = new DirectJdbcInputDStream[Row](ssc, jdbcParam)
directStream.foreachRDD(rdd => {
…//對(duì)數(shù)據(jù)進(jìn)行處理
})
2.4限流
假設(shè)當(dāng)前時(shí)間點(diǎn)到上次提取的時(shí)間點(diǎn)之間新增數(shù)據(jù)量太大,就會(huì)導(dǎo)致在新一次作業(yè)提交時(shí),上一次的作業(yè)仍然沒(méi)有完成,可能會(huì)因此造成作業(yè)積壓使得系統(tǒng)不穩(wěn)定。本文使用了基于規(guī)則的限流方法,綜合考慮集群處理能力以及間隔時(shí)間,可以配置化設(shè)置每次最大提取條數(shù)。如果當(dāng)前需要提取的數(shù)據(jù)條數(shù)大于最大提取條數(shù),則本次就只提取最大條數(shù),剩下的延時(shí)到下次再進(jìn)行提取。這樣做的好處是削減了峰流對(duì)系統(tǒng)造成的影響。
3測(cè)試分析
測(cè)試環(huán)境:VMware虛擬機(jī),處理器設(shè)置為4核心,2 GB內(nèi)存, 64位CentOS 6.5操作系統(tǒng),Spark 1.5.1,Oracle 11g。使用4臺(tái)虛擬機(jī)搭建成Spark集群,1臺(tái)為Master,3臺(tái)為Worker。數(shù)據(jù)庫(kù)表分別設(shè)置為20、40個(gè)字段,每次最大抽取記錄數(shù)分別設(shè)置為10 000、50 000、500 000。將抽取出來(lái)的數(shù)據(jù)寫成parquet格式的文件存儲(chǔ)到hdfs上。測(cè)試結(jié)果如表1所示。
4結(jié)束語(yǔ)
本文在基于數(shù)據(jù)庫(kù)觸發(fā)器記錄數(shù)據(jù)變化的基礎(chǔ)上,通過(guò)自己構(gòu)造DirectJdbcStream類提供Spark Streaming對(duì)數(shù)據(jù)庫(kù)的支持,達(dá)到準(zhǔn)實(shí)時(shí)從數(shù)據(jù)庫(kù)中抽取出增量數(shù)據(jù)的目的。并且可以對(duì)抽取出來(lái)的數(shù)據(jù)進(jìn)行過(guò)濾、清洗等操作,根據(jù)需求靈活地寫入到不同的目的庫(kù)。
參考文獻(xiàn)
?。?] 郭亮. 基于MD5與HASH的數(shù)據(jù)庫(kù)增量提取算法及其應(yīng)用[D]. 長(zhǎng)沙:湖南大學(xué),2013.
?。?] ZAHARIA M, CHOWDHURY M, DAS T, et al. Resilient distributed datasets: a fault tolerant abstraction for in memory cluster computing[C]. Usenix Conference on Networked Systems Design & Implementation, 2012, 70(2):141146.
?。?] DEAN J, GHEMAWAT S. MapReduce: simplified dataprocessing on large clusters[C]. USENIX Association OSDI′04: 6th Symposium on Operating Systems Design and Implementation, 2004:137149.
?。?] MARTIN O. Programming in scala[M]. California: Artima Press,2010.
[5] YADAV R. Spark cookbook[M]. UK: Packt Publishing Ltd, 2015.
[6] KARAU H. Learning spark[M]. America: O’Reilly Media, Inc. 2015.
[7] 梁剛. 企業(yè)大數(shù)據(jù)管理解決方案[J]. 微型機(jī)與應(yīng)用,2013,32(24):7 10,13.