Spark?GraphX?分布式圖處理框架圖算法詳解
正文
Spark GraphX是一個(gè)分布式圖處理框架,基于 Pregel
接口實(shí)現(xiàn)了常用的圖算法。
包括 PageRank、SVDPlusPlus、TriangleCount、 ConnectedComponents、LPA 等算法,以下通過(guò)具象化的圖實(shí)例理解相應(yīng)的算法用途。
Graphx圖結(jié)構(gòu)
Graphx
中的Graph有兩個(gè)RDD,一個(gè)是邊RDD
,一個(gè)是點(diǎn)RDD
。
此外,三元組其實(shí)就是(點(diǎn)、邊,點(diǎn))一個(gè)有效組合,由triplets
()接口獲取,triplets
()返回的結(jié)果是EdgeTriplet[VD,ED]
。
1. 最短路徑
最常見(jiàn)的路徑搜索算法(例如DFS
& BFS
、最短路徑、 最小生成樹(shù)、隨機(jī)游走等),最短路徑是最容易理解的圖算法,因?yàn)榇蠹以谏钪心軌驈V泛接觸到,如駕駛導(dǎo)航,外賣送餐路線等等。
路徑搜索算法建立在圖搜索算法的基礎(chǔ)上,用來(lái)探索節(jié)點(diǎn)之間的路徑。這些路徑從一個(gè)節(jié)點(diǎn)開(kāi)始,遍歷關(guān)系,直到到達(dá)目的地,Graphx采用了最短路徑算法Dijkstra
的原理。
示例數(shù)據(jù)
// 輸入一些邊數(shù)據(jù) val edgeSeq: Seq[(Int, Int)] = Seq((1, 2), (1, 5), (2, 3), (2, 5), (3, 4), (4, 5), (4, 6),(6, 9),(9, 11)).flatMap(e => Seq(e, e.swap)) val edges = op.sc.parallelize(edgeSeq).map { case (v1, v2) => (v1.toLong, v2.toLong) }
可視化數(shù)據(jù)
這是上述數(shù)據(jù)的圖形表示(雙向邊,無(wú)權(quán))
計(jì)算最短路徑
val graph_sp = Graph.fromEdgeTuples(edges, 1) val landmarks = Seq(1, 11).map(_.toLong) val results = ShortestPaths.run(graph_sp, landmarks).vertices.collect.map { case (v, spMap) => (v, spMap.mapValues(i => i)) }
全部結(jié)果打印:
println(results.mkString)
(1,Map(1 -> 0, 11 -> 5))
(2,Map(1 -> 1, 11 -> 5))
(3,Map(1 -> 2, 11 -> 4))
(4,Map(1 -> 2, 11 -> 3))
(5,Map(1 -> 1, 11 -> 4))
(6,Map(11 -> 2, 1 -> 3))
(9,Map(11 -> 1, 1 -> 4))
(11,Map(11 -> 0, 1 -> 5))
上述計(jì)算了圖中所有點(diǎn)到點(diǎn)1和點(diǎn)11的最短距離,(起點(diǎn)id,Map(目標(biāo)1 -> 最短路徑長(zhǎng)度,目標(biāo)2 -> 最短路徑長(zhǎng)度))。例如5,Map(1 -> 1, 11 -> 4)
說(shuō)明從5到1最短距離是1,5到11的最短距離是4。
2. 網(wǎng)頁(yè)排名
PageRank度量一個(gè)圖中每個(gè)頂點(diǎn)的重要程度,假定從u到v的一條邊代表v的重要性標(biāo)簽。例如,一個(gè)微博用戶被許多其它人粉,該用戶排名很高。GraphX帶有靜態(tài)和動(dòng)態(tài)PageRank的實(shí)現(xiàn)方法,這些方法在PageRank object
中。靜態(tài)的PageRank
運(yùn)行固定次數(shù)的迭代,而動(dòng)態(tài)的PageRank
一直運(yùn)行,直到收斂。
GraphX有一個(gè)我們可以運(yùn)行PageRank
的社交網(wǎng)絡(luò)數(shù)據(jù)集的簡(jiǎn)單數(shù)據(jù)。用戶集在graphx/data/users.txt
中,用戶之間的關(guān)系在graphx/data/followers.txt
中(Spark的源碼或編譯后文件里都包含)。
數(shù)據(jù)可視化
pagerank算法測(cè)試
先說(shuō)PageRank動(dòng)態(tài)實(shí)現(xiàn),以下調(diào)用就是動(dòng)態(tài)的,實(shí)際是調(diào)用runUntilConvergence()
不能指定迭代次數(shù)。參數(shù)0.0001是個(gè)容忍度,是在對(duì)圖進(jìn)行迭代過(guò)程中退出迭代的條件,而靜態(tài)的PageRank不可傳遞該參數(shù),但可以指定迭代次數(shù)【固定次數(shù),所以靜態(tài)】。
val graph: Graph[Int, Int] = GraphLoader .edgeListFile(op.sc, "followers.txt", canonicalOrientation = true, numEdgePartitions = 1) val ranks = graph.pageRank(0.0001).vertices.sortBy(_._2, ascending = false) ranks.take(5).foreach(println(_))
算法結(jié)果
(7,1.8138212152810693)
(2,1.0692956678358136)
(4,0.8759124087591241)
(6,0.8759124087591241)
(1,0.6825291496824343)
# join name
(odersky,1.8138212152810693)
(ladygaga,1.0692956678358136)
(justinbieber,0.8759124087591241)
(matei_zaharia,0.8759124087591241)
(BarackObama,0.6825291496824343)
二元組左側(cè)是頂點(diǎn)信息,右側(cè)是重要程度,也就是分?jǐn)?shù)越高排名越靠前。
這個(gè)結(jié)果有一些順序跟直觀感受不符,點(diǎn)7最重要毋庸置疑,點(diǎn)1的重要性應(yīng)該是大于點(diǎn)4的,但是結(jié)果不是這樣,那么數(shù)據(jù)集大一些會(huì)更好嗎??
personalizedPageRank()
方法還可以進(jìn)行個(gè)性化推薦,比如社交網(wǎng)絡(luò)中,給某用戶再推薦一個(gè)人,或者對(duì)于用戶商品的推薦中,用戶商品兩個(gè)實(shí)體可以形成一個(gè)圖,我們就可以根據(jù)具體的某個(gè)用戶來(lái)給他推薦一些商品。
3. 連通域(連通組件)
連通分量算法用其編號(hào)最小的頂點(diǎn)的 ID 標(biāo)記圖中的每個(gè)連通分量。例如,在社交網(wǎng)絡(luò)中,連接的組件可以近似集群。
加載圖測(cè)試連通域
這里的graph仍然是加載followers.txt數(shù)據(jù),spark自帶的有。
val cc: Graph[VertexId, Int] = graph.connectedComponents() println("連通結(jié)果展示++++++++:") cc.vertices.map(_.swap) .groupByKey() .map(_._2) .foreach(println)
結(jié)果展示(跟圖形觀察的結(jié)果是一致):
連通結(jié)果展示++++++++: CompactBuffer(4, 1, 2) CompactBuffer(6, 3, 7)
可以看到是2個(gè)域。結(jié)果圖數(shù)據(jù)本身不是這樣組織的,為了便于理解進(jìn)行了聚合。原始數(shù)據(jù)collect
回來(lái)是這樣:
Array((4,1), (1,1), (6,3), (3,3), (7,3), (2,1))
元組左側(cè)是頂點(diǎn),右側(cè)表示歸屬,這個(gè)結(jié)果符合預(yù)期。
生成圖測(cè)試
val g = Graph(sc.makeRDD((1L to 7L).map((_,""))), sc.makeRDD(Array(Edge(2L,5L,""), Edge(5L,3L,""), Edge(3L,2L,""), Edge(4L,5L,""), Edge(6L,7L,"")))) g.connectedComponents .vertices .map(_.swap) .groupByKey() .map(_._2) .foreach(println)
圖實(shí)例的形態(tài)展示
這樣的代碼便于自行組織一套圖數(shù)據(jù),按自己意思進(jìn)行修改,運(yùn)行上述代碼得到結(jié)果是:
CompactBuffer(1) CompactBuffer(2, 3, 4, 5) CompactBuffer(6, 7)
強(qiáng)連接網(wǎng)絡(luò)就是:在這個(gè)網(wǎng)絡(luò)中無(wú)論你從哪個(gè)頂點(diǎn)開(kāi)始,其他所有頂點(diǎn)都是可達(dá)的。
強(qiáng)連接域的計(jì)算
g.stronglyConnectedComponents(3) .vertices.map(_.swap) .groupByKey() .map(_._2) .filter(_.size > 1) .foreach(println)
過(guò)濾掉那些單點(diǎn)的域,那么強(qiáng)連接的計(jì)算結(jié)果是CompactBuffer(2, 3, 5)
。
4. 三角計(jì)數(shù)
當(dāng)一個(gè)頂點(diǎn)有兩個(gè)相鄰的頂點(diǎn)并且它們之間有一條邊時(shí),它就是三角形的一部分。需要注意的是,在計(jì)算社交網(wǎng)絡(luò)數(shù)據(jù)集的三角形計(jì)數(shù)時(shí),TriangleCount
需要邊的方向是規(guī)范的方向(srcId < dstId),并且圖通過(guò)Graph.partitionBy
分片過(guò)。
三角計(jì)數(shù)統(tǒng)計(jì)應(yīng)用場(chǎng)景:大規(guī)模的社區(qū)發(fā)現(xiàn),通過(guò)該算法可以做群體檢測(cè)。只要是跟大規(guī)模小團(tuán)體檢測(cè)方面該算法都可以很好的支持,算法是找出擁有三角形環(huán)關(guān)系的最多的頂點(diǎn)。
Triangle Count的算法思想如下:
- 計(jì)算每個(gè)結(jié)點(diǎn)的鄰結(jié)點(diǎn);
- 對(duì)通過(guò)每條邊的兩個(gè)頂點(diǎn)相聯(lián)的頂點(diǎn)的相鄰點(diǎn)集合計(jì)算交集,并找出交集中id大于前兩個(gè)結(jié)點(diǎn)id的結(jié)點(diǎn);
- 對(duì)每個(gè)結(jié)點(diǎn)統(tǒng)計(jì)Triangle總數(shù),注意只統(tǒng)計(jì)符合計(jì)算方向的Triangle Count。
代碼測(cè)試
val graph2 = graph.partitionBy(PartitionStrategy.RandomVertexCut) // Find the triangle count for each vertex val triCounts = graph2.triangleCount().vertices println(triCounts.collect().mkString("\n"))
開(kāi)頭先對(duì)圖graph進(jìn)行了分片得到graph2。
測(cè)試結(jié)果
這個(gè)意思是6,3,7頂點(diǎn)分別擁有1個(gè)三角環(huán),而其他頂點(diǎn)沒(méi)有,實(shí)際上正是6,3,7組成了三角。
(4,0)
(1,0)
(6,1)
(3,1)
(7,1)
(2,0)
5. 標(biāo)簽傳播算法(LPA)
Label Propagation,是一種基于圖的半監(jiān)督學(xué)習(xí)算法(Semi-supervised learning),應(yīng)用場(chǎng)景為:社區(qū)發(fā)現(xiàn)(Community detection)。社區(qū)發(fā)現(xiàn)的過(guò)程就是一種聚類的過(guò)程。主要是用于團(tuán)體檢測(cè),LPA能夠以接近線性復(fù)雜度去檢測(cè)一個(gè)大規(guī)模圖中的團(tuán)體結(jié)構(gòu),主要思想是給所有頂點(diǎn)中的密集連接組打上一個(gè)唯一標(biāo)簽,這些擁有相同標(biāo)簽的組就是所謂的團(tuán)體。
它不保證收斂,且迭代次數(shù)足夠多之后,所有聯(lián)通節(jié)點(diǎn)最終收斂為一個(gè)社區(qū)。
該算法也可以用于半監(jiān)督學(xué)習(xí)(大部分沒(méi)有標(biāo)簽,小部分有標(biāo)簽),給那些沒(méi)有標(biāo)簽的通過(guò)標(biāo)簽傳播算法進(jìn)行打標(biāo)簽。也可以應(yīng)用于風(fēng)控,對(duì)于通過(guò)已有風(fēng)險(xiǎn)評(píng)估的人,通過(guò)社交網(wǎng)絡(luò)去評(píng)估跟其有關(guān)系的人的風(fēng)險(xiǎn)。
基本思想
標(biāo)簽傳播算法的應(yīng)用場(chǎng)景是不重疊社區(qū)發(fā)現(xiàn),其基本思想是:將一個(gè)節(jié)點(diǎn)的鄰居節(jié)點(diǎn)的標(biāo)簽中數(shù)量最多的標(biāo)簽作為該節(jié)點(diǎn)自身的標(biāo)簽。給每個(gè)節(jié)點(diǎn)添加標(biāo)簽(label)以代表它所屬的社區(qū),并通過(guò)標(biāo)簽的“傳播”形成同一標(biāo)簽的“社區(qū)”結(jié)構(gòu)。簡(jiǎn)而言之,你的鄰居屬于哪個(gè)label最多,你就屬于哪個(gè)label。該算法的有點(diǎn)是收斂周期短,除了迭代次數(shù)無(wú)需任何先驗(yàn)參數(shù)(不需事先指定社區(qū)個(gè)數(shù)和大小),算法執(zhí)行過(guò)程中不需要計(jì)算任何社區(qū)指標(biāo)。
以上就是Spark GraphX 分布式圖處理框架圖算法詳解的詳細(xì)內(nèi)容,更多關(guān)于Spark GraphX 圖算法的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
負(fù)數(shù)與二進(jìn)制換轉(zhuǎn)方法
先談?wù)勈裁聪肫疝D(zhuǎn)載一篇這樣的文章。由于寫java已經(jīng)有一段時(shí)間了,在使用api上基本上沒(méi)有障礙,但是對(duì)有些基礎(chǔ)知識(shí)老是容易忘記,如二進(jìn)制和十進(jìn)制的一些轉(zhuǎn)換問(wèn)題。在此記錄一下,再次復(fù)習(xí)一下2013-02-02如何在vscode中正確使用正則表達(dá)式進(jìn)行文檔內(nèi)容的替換編輯
正則表達(dá)式是一種強(qiáng)大的模式匹配工具,它具有廣泛的應(yīng)用,包括數(shù)據(jù)清洗、文本處理、文件搜索等方面,這篇文章主要給大家介紹了關(guān)于如何在vscode中正確使用正則表達(dá)式進(jìn)行文檔內(nèi)容的替換編輯,需要的朋友可以參考下2023-12-12