GraphX 在圖數據庫 Nebula Graph 的圖計算實踐

不一樣來源的異構數據間存在着千絲萬縷的關聯,這種數據之間隱藏的關聯關係和網絡結構特性對於數據分析相當重要,圖計算就是以圖做爲數據模型來表達問題並予以解決的過程。java

圖計算實踐

1、背景

隨着網絡信息技術的飛速發展,數據逐漸向多源異構化方向發展,且不一樣來源的異構數據之間也存在的千絲萬縷的關聯,這種數據之間隱藏的關聯關係和網絡結構特性對於數據分析相當重要。但傳統關係型數據庫在分析大規模數據關聯特性時存在性能缺陷、表達有限等問題,所以有着更強大表達能力的圖數據受到業界極大重視,圖計算就是以圖做爲數據模型來表達問題並予以解決的過程。圖能夠融合多源多類型的數據,除了能夠展現數據靜態基礎特性以外,還可經過圖計算展現隱藏在數據之間的圖結構特性和點對關聯關係,成爲社交網絡、推薦系統、知識圖譜、金融風控、網絡安全、文本檢索等領域重要的分析手段。git

2、算法應用

爲了支撐大規模圖計算的業務需求,Nebula Graph 基於 GraphX 提供了 PageRankLouvain 社區發現的圖計算算法,容許用戶經過提交 Spark 任務的形式執行算法應用。此外,用戶也能夠經過 Spark Connector 編寫 Spark 程序調用 GraphX 自帶的其餘圖算法,如 LabelPropagation、ConnectedComponent 等。github

PageRank

PageRank 是谷歌提出的用於解決連接分析中網頁排名問題的算法,目的是爲了對互聯網中數以億計的網頁進行排名。算法

PageRank 簡介

美國斯坦福大學的 Larry Page 和 Sergey Brin 在研究網頁排序問題時採用學術界評判論文重要性的方法,即看論文的引用量以及引用該論文的論文質量,對應於網頁的重要性有兩個假設:shell

  1. 數量假設:若是一個網頁 A 被不少其餘網頁連接到,則該網頁比較重要;
  2. 質量假設:若是一個很重要的網頁連接到網頁 A,則該網頁的重要性會被提升。

並基於這兩個假設提出 PageRank 算法。數據庫

PageRank 應用場景

社交應用的類似度內容推薦

在對微博、微信等社交平臺進行社交網絡分析時,能夠基於 PageRank 算法根據用戶一般瀏覽的信息以及停留時間實現基於用戶的類似度的內容推薦;apache

分析用戶社交影響力

在社交網絡分析時根據用戶的 PageRank 值進行用戶影響力分析;安全

文獻重要性研究

根據文獻的 PageRank 值評判該文獻的質量,PageRank 算法就是基於評判文獻質量的想法來實現設計。微信

此外 PageRank 在數據分析和挖掘中也有不少的應用。網絡

算法思路

GraphX 的 PageRank 算法是基於 Pregel 計算模型的,該算法流程包括 3 步驟:

  1. 爲圖中每一個節點(網頁)設置一個一樣的初始 PageRank 值;
  2. 第一次迭代:沿邊發送消息,每一個節點收到全部關聯邊上對點的信息,獲得一個新的 PageRank 值;
  3. 第二次迭代:用這組新的 PageRank 按不一樣算法模式對應的公式造成節點本身新的 PageRank。

Louvain 社區發現

Louvain 是用來進行社會網絡挖掘的社區發現算法,屬於圖的聚類算法。

Louvain 算法介紹

Louvain 是基於模塊度(Modularity)的社區發現算法,經過模塊度來衡量一個社區的緊密程度。若是一個節點加入到某一社區中會使得該社區的模塊度相比其餘社區有最大程度的增長,則該節點就應當屬於該社區。若是加入其它社區後沒有使其模塊度增長,則留在本身當前社區中。

模塊度

模塊度公式

模塊度 Q 的物理意義:社區內節點的連邊數與隨機狀況下的邊數之差,定義函數以下:

其中

:節點 i 和節點 j 之間邊的權重 :全部與節點 i 相連的邊的權重之和 :節點 i 所屬的社區 : 圖中全部邊的權重之和

模塊度公式變形

在此公式中,只有節點 i 和節點 j 屬於同一社區,公式纔有意義,因此該公式是衡量的某一社區內的緊密度。對於該公式的簡化變形以下:

表示: 社區 c 內的邊的權重之和 表示: 全部與社區 c 內節點相連的邊的權重之和(由於 i 屬於社區 c)包括社區內節點與節點 i 的邊和社區外節點與節點 i 的邊。 表示: 全部與社區 c 內節點相連的邊的權重之和(由於 j 屬於社區 c)包括社區內節點與節點 j 的邊和社區外節點與節點 j 的邊。 代替 。(即社區 c 內邊權重和 + 社區 c 與其餘社區連邊的權重和)

求解模塊度變化

在 Louvain 算法中不須要求每一個社區具體的模塊度,只須要比較社區中加入某個節點以後的模塊度變化,因此須要求解 △Q。

將節點 i 分配到某一社區中,社區的模塊度變化爲:

其中

: 社區內全部節點與節點 i 連邊權重之和(對應新社區的實際內部權重和乘以 2,由於 對於社區內全部的頂點 i,每條邊其實被計算了兩次) : 全部與節點 i 相連的邊的權重之和 故實現算法時只需求 便可。

Louvain 應用場景

  • 金融風控

在金融風控場景中,能夠根據用戶行爲特徵進行團伙識別;

  • 社交網絡

能夠基於網絡關係中點對之間關聯的廣度和強度進行社交網絡劃分;對複雜網絡分析、電話網絡分析人羣之間的聯繫密切度;

  • 推薦系統

基於用戶興趣愛好的社區發現,能夠根據社區並結合協同過濾等推薦算法進行更精確有效的個性化推薦。

Louvain 算法思路

Louvain 算法包括兩個階段,其流程就是這兩個階段的迭代過程。

階段一:不斷地遍歷網絡圖中的節點,經過比較節點給每一個鄰居社區帶來的模塊度的變化,將單個節點加入到可以使 Modularity 模塊度有最大增量的社區中。 (好比節點 v 分別加入到社區 A、B、C 中,使得三個社區的模塊度增量爲-1, 1, 2, 則節點 v 最終應該加入到社區 C 中)

階段二:對第一階段進行處理,將屬於同一社區的頂點合併爲一個大的超點從新構造網絡圖,即一個社區做爲圖的一個新的節點。此時兩個超點之間邊的權重是兩個超點內全部原始頂點之間相連的邊權重之和,即兩個社區之間的邊權重之和。

下面是對第一二階段的實例介紹。

Louvain 社區算法

第一階段遍歷圖中節點加入到其所屬社區中,獲得中間的圖,造成四個社區;

第二節點對社區內的節點進行合併成一個超級節點,社區節點有自連邊,其權重爲社區內部全部節點間相連的邊的權重之和的 2 倍,社區之間的邊爲兩個社區間頂點跨社區相連的邊的權重之和,如紅色社區和淺綠色社區之間經過(8,11)、(10,11)、(10,13)相連,因此兩個社區之間邊的權重爲 3。

注:社區內的權重爲全部內部結點之間邊權重的兩倍,由於 Kin 的概念是社區內全部節點與節點 i 的連邊和,在計算某一社區的 Kin 時,實際上每條邊都被其兩端的頂點計算了一次,一共被計算了兩次。

整個 Louvain 算法就是不斷迭代第一階段和第二階段,直到算法穩定(圖的模塊度再也不變化)或者到達最大迭代次數。

3、算法實踐

演示環境

  • 三臺虛擬機,環境以下:
    • Cpu name:Intel(R) Xeon(R) Platinum 8260M CPU @ 2.30GHz
    • Processors:32
    • CPU Cores:16
    • Memory Size:128G
  • 軟件環境
    • Spark:spark-2.4.6-bin-hadoop2.7 三個節點集羣
    • yarn V2.10.0:三個節點集羣
    • Nebula Graph V1.1.0:分佈式部署,默認配置

測試數據

  1. 建立圖空間
CREATE SPACE algoTest(partition_num=100, replica_factor=1);
  1. 建立點邊 Schema
CREATE TAG PERSON()
CREATE EDGE FRIEND(likeness double);
  1. 導入數據

利用 Exchange 工具將數據離線導入 Nebula Graph。

  1. 測試結果

Spark 任務的資源分配爲 --driver-memory=20G --executor-memory=100G --executor-cores=3

  • PageRank 在一億數據集上的執行時間爲 21min(PageRank 算法執行時間)
  • Louvain 在一億數據集上的執行時間爲 1.3h(Reader + Louvain 算法執行時間)

如何使用 Nebula Graph 的算法

  1. 下載 nebula-algorithm 項目並打成 jar 包
$ git clone git@github.com:vesoft-inc/nebula-java.git
$ cd nebula-java/tools/nebula-algorithm
$ mvn package -DskipTests
  1. 配置項目中的 src/main/resources/application.conf 
{
  # Spark relation config
  spark: {
    app: {
        # not required, default name is the algorithm that you are going to execute.
        name: PageRank

        # not required
        partitionNum: 12
    }

    master: local

    # not required
    conf: {
        driver-memory: 8g
        executor-memory: 8g
        executor-cores: 1g
        cores-max:6
    }
  }

  # Nebula Graph relation config
  nebula: {
    # metadata server address
    addresses: "127.0.0.1:45500"
    user: root
    pswd: nebula
    space: algoTest
    # partition specified while creating nebula space, if you didn't specified the partition, then it's 100.
    partitionNumber: 100
    # nebula edge type
    labels: ["FRIEND"]

    hasWeight: true
    # if hasWeight is true,then weightCols is required, and weghtCols' order must be corresponding with labels.
    # Noted: the graph algorithm only supports isomorphic graphs,
    #        so the data type of each col in weightCols must be consistent and all numeric types.
    weightCols: [「likeness」]
  }

  algorithm: {
    # the algorithm that you are going to execute,pick one from [pagerank, louvain]
    executeAlgo: louvain
    # algorithm result path
    path: /tmp

    # pagerank parameter
    pagerank: {
        maxIter: 20
        resetProb: 0.15  # default 0.15

    }

    # louvain parameter
    louvain: {
        maxIter: 20
        internalIter: 10
        tol: 0.5
   }
  }
}
  1. 確保用戶環境已安裝 Spark 並啓動 Spark 服務

  2. 提交 nebula-algorithm 應用程序:

spark-submit --master xxx --class com.vesoft.nebula.tools.algorithm.Main /your-jar-path/nebula-algorithm-1.0.1.jar -p /your-application.conf-path/application.conf

若是你對上述內容感興趣,歡迎用 nebula-algorithm 試試^^

References

做者有話說:Hi,我是安祺,Nebula Graph 研發工程師,若是你對本文有任何疑問,歡迎來論壇和我交流:https://discuss.nebula-graph.com.cn/

推薦閱讀

相關文章
相關標籤/搜索