一致性Hash算法在資料庫分表中的實踐

一致性Hash算法在數據庫分表中的實踐

Hash算法

最近有一個項目,其中某個功能單表數據在可預估的未來達到了億級,初步估算在90億左右。與同事詳細討論後,決定採用一致性Hash算法來完成數據庫的自動擴容和數據遷移。整個程序細節由我同事完成,我只是將其理解併成文,供有相同問題的同行參考。

參看此文的兄弟,默認各位已經熟悉一致性hash算法了。此文僅僅闡述代碼細節,實現語言為Java。

項目背景

  1. 項目是一個實驗室項目
  2. 其中有一個表叫做試驗表,用於存儲車型的試驗數據,每個試驗大概有6000條數據
  3. 總計初期約有2萬個車型,每個車型初期包含超過50個試驗。後期還會動態增長
  4. 試驗表中的數據僅需要根據車型試驗ID能取出來即可,沒有其他更復雜的業務邏輯

方案決策

項目正式上線初期,數據量不會直接爆發式增長到90億,需要時間上的積累(逐步做實驗),最終可能達到90億數據,甚至超過90億數據。

按照我們實際瞭解情況,oracle存儲數據量達到1千萬的時候,性能擅可。而Oracle官方的說法,如單表存儲1g有分區(大致500萬數據),查詢效率非常高。而試驗表中僅四個字段,每條數據數據量較小。所以我們最終決定以1000萬為節點,水平拆表。當表數據達到1千萬時,即增加下一波表。進行數據自動遷移。

按照90億的總量,1000萬數據一個表的劃分,最終大致會產生900個左右的表。所以我們最終使用了4個數據庫。1個存儲其他業務模塊的表,3個存儲此大數據表。每個數據庫大致有300張表。性能上和數量上都可達到我們的要求。

相關表結構

試驗信息表(EXPERIMENT_MESSAGE),掛接車型和試驗的關係。試驗數據表(EXPERIMENT_DATA),存儲試驗數據

試驗信息表:

字段含義ID主鍵,採用UUID生成EXPERIMENT_ID試驗表中的IDCAR_ID車型表中的ID...其餘數十個字段省略

試驗數據表:

字段含義ID主鍵,採用UUID生成EXPERIMENT_MESSAGE_ID對應的實驗信息idX_VALUE試驗數據X值Y_VALUE試驗數據Y值

我們採用作一致性hash的key,就是試驗數據表中的EXPERIMENT_MESSAGE_ID字段。也就是說,每個試驗數據表,不存則以,存則一次性大致有6000條數據。取同理。

一致性Hash算法實現

一致性Hash算法的hash部分,採用了著名的ketama算法。在此,我們不多討論ketama算法的細節,若各位有興趣,請查閱ketama算法

 public long hash(String key) {
if (md5 == null) {
try {
md5 = MessageDigest.getInstance("MD5");
} catch (NoSuchAlgorithmException e) {
throw new IllegalStateException("no md5 algorythm found");
}
}
md5.reset();
md5.update(key.getBytes());
byte[] bKey = md5.digest();
long res = ((long) (bKey[3] & 0xFF) << 24) |
((long) (bKey[2] & 0xFF) << 16) |
((long) (bKey[1] & 0xFF) << 8) |
(long) (bKey[0] & 0xFF);
return res & 0xffffffffL;
}

有了Hash的算法,接下來就要構造Hash環了。Hash環採用的SortedMap數據結構實現。

private final SortedMap circle = new TreeMap();

其中添加節點和移除節點部分,需要根據hash算法得到節點在環上的位置,具體代碼如下:

 /**
* 添加虛擬節點
* numberOfReplicas為虛擬節點的數量,初始化hash環的時候傳入,我們使用300個虛擬節點
* @param node
*/
public void add(T node) {
for (int i = 0; i < numberOfReplicas; i++) {
circle.put(hashFunction.hash(node.toString() + i), node);
}
}
/**
* 移除節點
* @param node
*/
public void remove(T node) {
for (int i = 0; i < numberOfReplicas; i++) {
circle.remove(hashFunction.hash(node.toString() + i));
}
}

而hash環中得到節點部分比較特殊,根據一致性hash算法的介紹,得到hash環中的節點,實際上是計算出的hash值順時針找到的第一個節點。

 /**
* 獲得一個最近的順時針節點
* @param key 為給定鍵取Hash,取得順時針方向上最近的一個虛擬節點對應的實際節點

* @return
*/
public T get(Object key) {
if (circle.isEmpty()) {
return null;
}
long hash = hashFunction.hash((String) key);
if (!circle.containsKey(hash)) {
//返回此映射的部分視圖,其鍵大於等於 hash
SortedMap tailMap = circle.tailMap(hash);
hash = tailMap.isEmpty() ? circle.firstKey() : tailMap.firstKey();
}
return circle.get(hash);
}

單表拆分實踐

上面完成了一致性hash算法的實現,包含了hash算法和hash環的實現。接下來就要處理具體業務中,如何使用這個hash環和算法了。

我們業務中,主要操作這張表的數據,也就是增刪查。然後我們數據庫拆分成了3個,所以需要增刪查的操作基本一致,都是先通過一致性hash得到庫,再通過一致性hash得到表。

獲取數據庫名的操作如下,獲取到數據庫後,根據數據庫名到對應的連接池中獲取連接。

 /**
* 根據試驗信息id獲取其所在庫名
* DatabaseType為我們數據的枚舉

* @return 數據庫的名稱
**/
private String getDataBase(String experimentMessageId) {
//獲取數據源
DatabaseType[] databasetype = DatabaseType.values();
List dataBaselist = new ArrayList<>();
Map map = new HashMap<>();
for (DatabaseType d:databasetype) {
if (!d.equals(DatabaseType.KC)) {
dataBaselist.add(d.toString());
map.put(d.toString(), d);
}
}
//獲取數據源hash
ConsistentHash dataBaseCon = getConsistentHash(dataBaselist);
//獲取id所在數據源
String dataBase = dataBaseCon.get(experimentMessageId);
return dataBase;
}

獲取表名的操作如下,獲取到數據庫後,在對應的數據庫中找到需要的表,再從該表中查詢數據。

 /**
* 根據試驗信息id獲取其試驗數據所在表
* @return
**/
public String getTableName(String experimentMessageId) {
String dataBase = getDataBase(experimentMessageId);
//查詢所有試驗數據表
List tables = experimentDataEODao.queryTbaleNames(dataBase, tableName);
ConsistentHash consistentHash = getConsistentHash(tables);
String tableName = consistentHash.get(experimentMessageId);

return tableName;
}

剩下的增刪改操作和平常一致,在此不多贅述。

數據遷移實踐

一致性hash勢必涉及到數據遷移問題,我們採取的數據遷移方式為定時任務,針對每個數據庫在每天夜裡全量掃描一次。檢查是否有數據量超過1000萬的表,若存在這樣的表,就把現有的表數量double。

數據遷移只會在同庫之間遷移,不會涉及跨數據庫的情況。

此方案為初步方案,後續會改進的更加智能,根據表的數量,增加不同數量的表。而不是簡單的把表數量翻倍。

表創建後,將需要遷移的表數據逐個遷移。

在連接到數據源後,我們做了如下事情進行數據遷移

1.獲取庫中所有的表

 List tables = getTables(connection, p, d.toString());

2.遍歷表,檢查表中數據是否超過邊界線(我們為1000萬)

 for (int i = 0; i < tables.size(); i++) {
//查詢表內數據量
int num = countByTableName(connection, p, tables.get(i));
//finalNum為邊界值,此處為1000萬
if (num > finalNum) {
……
}
……
}

3.根據所有的表計算現有的虛擬節點

ConsistentHash consistentHashOld = getConsistentHash(tables);

4.把表加倍

List tablesNew = deepCopy(tables); //注意一定要採用深複製
int tableSize = tablesNew.size();
for (int y = 0; y < tableSize; y++) {
String tableNameNew = tableName + (tablesNew.size() + 1);
//創建表
createTable(connection, p, d.toString(), tableNameNew);
tablesNew.add(tableNameNew);
tableDelete.add(tableNameNew);
}

5.計算加倍後的虛擬節點

ConsistentHash consistentHashNew = getConsistentHash(tablesNew);

6.數據遷移

for (int z = 0; z < tableSize; z++) {
String tableNameOld = tablesNew.get(z);
//查詢試驗信息id不重複的試驗數據信息
List disData = selectExperimentIdDis(connection, p, tableNameOld);
List deleteList = new LinkedList<>();
for (String experimentId : disData) {
//如果數據hash計算 原所在表與新建表之後不一致,執行轉移
if (!consistentHashNew.get(experimentId).equals(consistentHashOld.get(experimentId))) {
//新增到新表數據
insertHash(connection, p, experimentId, consistentHashOld.get(experimentId),
consistentHashNew.get(experimentId));
//刪除數據集合
deleteList.add(experimentId);
//刪除舊錶數據
final int defaultDelNum = 1000;
if (deleteList.size() == defaultDelNum) {
deleteInbatch(connection, p, deleteList, tableNameOld);
deleteList.clear();
}
}
}
//刪除舊錶數據
if (deleteList.size() > 0) {
deleteInbatch(connection, p, deleteList, tableNameOld);
}
}

總結

以上為我們所做的一致性hash實踐,其中還存在很多問題,比如遷移過程單線程導致遷移較慢、自動擴容機制不智能、遷移過程中數據訪問不穩定等情況。

我們將會在後續的開發中逐步進行完善改進。

以上就是我們針對一致性hash在oracle分表中的實踐

加Java架構師進階交流群獲取Java工程化、高性能及分佈式、高性能、深入淺出。高架構。性能調優、Spring,MyBatis,Netty源碼分析和大數據等多個知識點高級進階乾貨的直播免費學習權限 都是大牛帶飛 讓你少走很多的彎路的 群號是:338549832 對了 小白勿進 最好是有開發經驗

注:加群要求

1、具有工作經驗的,面對目前流行的技術不知從何下手,需要突破技術瓶頸的可以加。

2、在公司待久了,過得很安逸,但跳槽時面試碰壁。需要在短時間內進修、跳槽拿高薪的可以加。

3、如果沒有工作經驗,但基礎非常紮實,對java工作機制,常用設計思想,常用java開發框架掌握熟練的,可以加。

4、覺得自己很牛B,一般需求都能搞定。但是所學的知識點沒有系統化,很難在技術領域繼續突破的可以加。

5.阿里Java高級大牛直播講解知識點,分享知識,多年工作經驗的梳理和總結,帶著大家全面、科學地建立自己的技術體系和技術認知!


分享到:


相關文章: