前半有序的大資料排序
最近碰到這麼一個案例,情況可以簡化總結成這樣:資料庫中有表T,其中有兩個重要的欄位a和b,a是一個時間戳,精確到秒;b是使用者號;其它欄位用來表示使用者b在時刻a發生的事件屬性。
現在任務是:把資料按a,b排序匯出。簡單來講,就是把SELECT * FROM T ORDER BY a,b的結果集寫出到檔案。
但是,這個T表有幾十億條記錄,這個SQL發出去之後,資料庫就象死了一樣,一個多小時都沒有任何反應。
這也難怪,幾十億記錄的大排序確實非常慢,上T的資料量記憶體裝不下,而外存排序需要分段讀入資料,在記憶體將每段排序後再快取到硬碟,然後將這些快取資料一起再歸併。這樣,必須把所有資料都遍歷過一遍且分段排序後才能開始輸出。
還有什麼別的辦法麼?
通用的大排序可以說已經被全世界研究到極致了,再想出一個更優的辦法幾乎沒有可能性了。但是,如果我們能找到這些資料的一些可得特徵,說不定就能有辦法了。
瞭解到一些業務資訊後,我們發現這批資料有這樣一些特徵:
1. 資料是按發生時刻(也就是a)為次序插入的,這樣物理儲存次序也就會接近於按a有序,而且資料庫已經為欄位a建好了索引;
2. 從某個起始時刻到終止時刻,幾乎每一秒都會有資料插入;
3. 資料按時間分佈比較平均,大概每秒數萬條,沒有某一秒的資料量特別多;
利用這些特徵,我們可以設計這樣的演算法(用SPL寫成),其中start,end分別是資料的起止時刻。
A |
B |
|
1 |
for interval@s(start,end)+1 |
=elapse@s(start,A1-1) |
2 |
=db.query(“SELECT * FROM T WHERE a=?”,B1) |
|
3 |
=B2.sort(b) |
|
4 |
>outputfile.export@a(B3) |
基本邏輯是:迴圈所有的秒,從資料庫取出某一秒的記錄按b排序後再寫出到檔案。因為資料庫為a建有索引,而資料也接近於按a有序儲存,用索引取數就非常快。每一秒內的資料量並不大,可以在記憶體中排序,速度很快。容易證明這個演算法返回的結果集就是按a,b有序的,這樣就不需要快取資料就可以完成這個大排序了。
這個演算法執行後立即就有資料開始輸出,數小時內就完成了按序匯出資料的任務,之所以需要數小時,主要還是從資料庫中取數以及寫入檔案的時間(幾十億行和上T的資料量),排序本身幾乎沒有佔用時間。
針對這批資料,我們還有一個任務:想知道欄位a,b是否可以用作T的主鍵,也就是說欄位a,b的取值在T表是否是唯一的。
本來用SQL做這個判斷也很簡單,只要看看
SELECT COUNT(*) FROM T
和
SELECT COUNT(*) FROM (SELECT a,b FROM T GROUP BY a,b)
是否相等就可以了(有些資料庫不支援COUNT(DISTINCT a,b)寫法,這裡寫成子查詢形式)。
COUNT(*)容易算,但面對數十億行的大資料做GROUP BY運算,其方法和外存排序是差不多的,成本也差不多,也是跑了一個多小時沒動靜。
但是,如果我們利用上述特徵,就很容易計算出這個值:
A |
B |
|
1 |
for interval@s(start,end)+1 |
=elapse@s(start,A1-1) |
2 |
=db.query@1(“SELECT COUNT(DISTINCT b) FROM T WHERE a=?”,B1) |
|
3 |
=@+B2 |
類似地,迴圈每一秒,針對每一條記錄一個COUNT(DISTINCT B),然後都加起來就是我們要的答案了(容易證明這個演算法的正確性)。這段程式碼幾分鐘就完成了運算(和上例相比,這裡不匯出也就不需要取出明細資料了,也不必寫檔案,而且還能平行計算,不象上例中要有序寫出就只能序列)。
細心的讀者可能要問,這兩個例子都是講如何利用索引來快速計算,為什麼本文標題要叫“前半有序”呢?
實際上我們就是利用了這批資料已經有的次序資訊。這兩個問題的關鍵點都是需要按a,b排序,而在索引的作用下,這批資料看起來已經對a有序了,也就是待排序欄位中的前一部分欄位已有序了。而如果前面欄位相同時的記錄數都沒有大到記憶體放不下的地步,那麼就可以不使用快取實現大排序了。
如果資料已經儲存在可以保持次序的檔案中,則這個方法的適應面會更寬泛一些,不需要事先知道a的起止時刻並迴圈每一秒,程式碼也會更簡單些。
假如資料檔案T中按a的次序寫入了T表的記錄,則上面的兩個問題的演算法可以分別寫出來是這樣:
A |
B |
|
1 |
for file(T).cursor();a |
=A1.sort(b) |
2 |
>outputfile.export@a(B1 |
A |
B |
|
1 |
for file(T).cursor();a |
=@+A1.id(b).len() |
SPL中提供了針對遊標的有序取出方法,上面兩段程式碼中A1格的意思是針對檔案T的資料遊標迴圈,每次讀到欄位a的值發生變化時則進入迴圈體,然後再讀下一批a相同的記錄,…。
基於檔案的運算比上述使用索引從資料庫取數的效果又好了數倍。而且這幾段程式碼對記憶體佔用也非常少。本來大排序是個很耗用記憶體的動作,因為後一步歸併的效能嚴重依賴於分段的數量,要減少分段,就要讓每一段儘量大,所以記憶體越大的效能就越好。而利用前半有序的特徵後,只要一點點記憶體(本例中只要能裝入數萬行記錄)就可以高速完成運算了。
最後再溫習一下我們的觀點:效能優化要因地制宜,根據資料和運算的特徵想辦法。
我們不能解決通用的大排序問題,但在特定場合下卻能設計出好演算法提高效能。而資料庫過於透明,看起來程式設計師不用操心了,但資料庫並沒有那麼智慧,經常不會利用資料特徵來自動優化。而且,在SQL體系下,即使人為想出好演算法,也幾乎無法實現。
原文釋出時間為:2018-11-13
本文作者:蔣步星
本文來自雲棲社群合作伙伴“資料蔣堂”,瞭解相關資訊可以關注“資料蔣堂”。
相關文章
- 效能最佳化技巧:前半有序時的排序排序
- Redis五大資料型別之 Zset(有序集合)Redis大資料資料型別
- 順序表有序插入資料
- 大資料集報表點選表頭排序大資料排序
- Redis 設計與實現 10:五大資料型別之有序集合Redis大資料資料型別
- MySQL-排序資料MySql排序
- Oracle Goldengate是如何保證資料有序和確保資料不丟失的?OracleGo
- 資料結構高階--八大排序彙總資料結構排序
- MySQL 是如何實現資料的排序的?MySql排序
- 資料結構與排序資料結構排序
- 資料庫排序查詢資料庫排序
- .net 對JSON資料排序JSON排序
- 資料結構(python) —— 【18排序: 桶排序】資料結構Python排序
- 什麼叫大資料 大資料的概念大資料
- 大資料如何採集資料?大資料的資料從何而來?大資料
- 送你一個Python 資料排序的好方法Python排序
- js 陣列返回,資料排序JS陣列排序
- 資料排序_麥克機試排序
- 資料結構--排序--插入排序--python語言描述資料結構排序Python
- 大資料是什麼?大資料的趨勢?大資料
- 【資料結構與演算法】非比較排序(計數排序、桶排序、基數排序)資料結構演算法排序
- (戀上資料結構筆記):計數排序、基數排序 、桶排序資料結構筆記排序
- mysql的中文資料按拼音排序的2個方法MySql排序
- 大資料時代的資料治理!大資料
- javascript: 帶分組資料的Table表頭排序JavaScript排序
- 資料結構32:選擇排序資料結構排序
- 資料結構之計數排序資料結構排序
- GBase8a資料排序優化排序優化
- 資料結構第10章 排序資料結構排序
- 資料結構實驗之連結串列六:有序連結串列的建立資料結構
- 資料重整:用Java實現精準Excel資料排序的實用策略JavaExcel排序
- 大資料MongoDB之mgo驅動如何對查詢結果進行排序(正序逆序多欄位排序)?大資料MongoDB排序
- 走進大資料,感受大資料大資料
- 到底什麼才是大資料技術?大資料的概念?大資料
- 大資料教程分享實用的大資料之陣列大資料陣列
- 【大資料】科普一下大資料的那些事兒大資料
- 【PHP資料結構】插入類排序:簡單插入、希爾排序PHP資料結構排序
- 【工業大資料】工廠大資料之資料來源分析;如何挖掘並駕馭大資料的價值,成為“大資料企業”?大資料