Spark GraphX聚合求和

时间:2017-01-03 20:38:40

标签: scala apache-spark spark-graphx

我试图在spark graphx图中计算节点值的总和。简而言之,图形是一棵树,顶部节点(根)应该是所有孩子和他们的孩子的总和。我的图表实际上是一棵看起来像这样的树, 预期求和值应为1850

                                     +----+
                     +--------------->    |  VertexID 14
                     |               |    |  Value: 1000
                 +---+--+            +----+
    +------------>      | VertexId 11
    |            |      | Value:     +----+
    |            +------+ Sum of 14 & 24  |  VertexId 24
+---++                +-------------->    |  Value: 550
|    | VertexId 20                   +----+
|    | Value:
+----++Sum of 11 & 911
      |
      |           +-----+
      +----------->     | VertexId 911
                  |     | Value: 300
                  +-----+

第一次尝试就是这样:

val vertices: RDD[(VertexId, Int)] =
      sc.parallelize(Array((20L, 0)
        , (11L, 0)
        , (14L, 1000)
        , (24L, 550)
        , (911L, 300)
      ))

  //note that the last value in the edge is for factor (positive or negative)
    val edges: RDD[Edge[Int]] =
      sc.parallelize(Array(
        Edge(14L, 11L, 1),
        Edge(24L, 11L, 1),
        Edge(11L, 20L, 1),
        Edge(911L, 20L, 1)
      ))

    val dataItemGraph = Graph(vertices, edges)


    val sum: VertexRDD[(Int, BigDecimal, Int)] = dataItemGraph.aggregateMessages[(Int, BigDecimal, Int)](
      sendMsg = { triplet => triplet.sendToDst(1, triplet.srcAttr, 1) },
      mergeMsg = { (a, b) => (a._1, a._2 * a._3 + b._2 * b._3, 1) }
    )

    sum.collect.foreach(println)

返回以下内容:

(20,(1,300,1))
(11,(1,1550,1))

它为顶点11做了总和,但它没有滚动到根节点(顶点20)。我错过了什么,或者有更好的方法吗?当然,树可以是任意大小的,每个顶点可以有任意数量的子边。

1 个答案:

答案 0 :(得分:6)

鉴于图形是有针对性的(如你所示的那样),应该可以编写一个Pregel程序来完成你所要求的:

val result = 
 dataItemGraph.pregel(0, activeDirection = EdgeDirection.Out)(
  (_, vd, msg) => msg + vd, 
  t => Iterator((t.dstId, t.srcAttr)), 
  (x, y) => x + y
 )

 result.vertices.collect().foreach(println)

// Output is:
// (24,550)
// (20,1850)
// (14,1000)
// (11,1550)
// (911,300)

我正在使用EdgeDirection.Out,因此消息只是从下到上发送(否则我们会进入无限循环)。