迈克的d3js Cluster Force Layout IV块

时间:2017-01-25 08:25:22

标签: math d3.js visualization data-visualization

我是d3js的新手,我刚开始。

我正在尝试Mike在他的一个块中编写的集群布局示例。 https://bl.ocks.org/mbostock/7882658

我用我的代码在我的机器上工作但我真的不喜欢盲目地复制代码而不理解它。

然而,我很难理解'cluster()'和'collide()'函数背后的数学以及它们如何运作。

有人可以解释一下吗?谢谢你的帮助!!

1 个答案:

答案 0 :(得分:1)

让我们看一下每种方法,我会尽可能地评论它。

<强>群集

首先是来电者:

 function tick(e) {
  node
    .each(cluster(10 * e.alpha * e.alpha)) //for each node on each tick call function returned by cluster function
                                           //pass in alpha cooling parameter to collide
  ...

我不会在这里重复关于tick事件如何工作的解释。 documentation很清楚。

功能:

// returns a closure wrapping the cooling
// alpha (so it can be used for every node on the tick)
function cluster(alpha) {
  return function(d) { // d here is the datum on the node 
    var cluster = clusters[d.cluster]; // clusters is a hash-map, the key is an index of the 10 clusters, the value is an object where d.cluster is the center node in that cluster
    if (cluster === d) return;  // if we are on the center node, do nothing
    var x = d.x - cluster.x, // distance on x of node to center node
        y = d.y - cluster.y, // distance on y of node to center node
        l = Math.sqrt(x * x + y * y), // distance of node to center node (Pythagorean theorem)
        r = d.radius + cluster.radius; // radius of node, plus radius of center node (the center node is always the largest one in the cluster)
    if (l != r) { // if the node is not adjacent to the center node
      l = (l - r) / l * alpha; //find a length that is slightly closer, this provides the illusion of it moving towards the center on each tick
      d.x -= x *= l; // move node closer to center node
      d.y -= y *= l; 
      cluster.x += x; // move center node closer to node
      cluster.y += y;
    }
  };
}

<强>碰撞

碰撞功能有点复杂。在我们深入研究之前,您需要了解QuadTree是什么以及为什么Bostock正在使用它。如果你想确定两个元素是否碰撞,那么天真算法就是循环外部和内部的元素,以便将每个元素与每个元素进行比较。当然,这在计算上非常昂贵,特别是在每个滴答声中。这是QuadTrees试图解决的问题:

  

四叉树递归地将二维空间划分为正方形,将每个正方形划分为四个大小相等的正方形。每个不同的点存在于唯一的叶节点中;重合点由链表表示。四叉树可以加速各种空间操作,例如用于计算多体力的Barnes-Hut近似,碰撞检测和搜索附近点。

这是什么意思?首先,看看这个excellent explanation。用我自己的简化词来表示这意味着:取一个二维空间并将其划分为四个象限。如果任何象限包含4个或更少的节点停止。如果象限包含四个以上的节点,请将其再次划分为四个象限。重复此操作,直到每个象限/子象限包含4个或更少的节点。现在,当我们寻找碰撞时,我们的内部循环不再循环节点,而是象限。如果象限没有碰撞,则移动到下一个。这是一个很大的优化。

现在进入代码:

// returns a closure wrapping the cooling
// alpha (so it can be used for every node on the tick)
// and the quadtree
function collide(alpha) {
  // create quadtree from our nodes
  var quadtree = d3.geom.quadtree(nodes);
  return function(d) { // d is the datum on the node
    var r = d.radius + maxRadius + Math.max(padding, clusterPadding), // r is the radius of the node circle plus padding
        nx1 = d.x - r, // nx1, nx2, ny1, ny2 are the bounds of collision detection on the node
        nx2 = d.x + r,
        ny1 = d.y - r,
        ny2 = d.y + r;
    quadtree.visit(function(quad, x1, y1, x2, y2) { // visit each quadrant
      if (quad.point && (quad.point !== d)) { // if the quadrant is a point (a node and not a sub-quadrant) and that point is not our current node
        var x = d.x - quad.point.x, // distance on x of node to quad node
            y = d.y - quad.point.y, // distance on y of node to quad node
            l = Math.sqrt(x * x + y * y), // distance of node to quad node (Pythagorean theorem)
            r = d.radius + quad.point.radius + (d.cluster === quad.point.cluster ? padding : clusterPadding); // radius of node in quadrant
        if (l < r) { // if there is a collision
          l = (l - r) / l * alpha; // re-position nodes
          d.x -= x *= l;  
          d.y -= y *= l;
          quad.point.x += x;
          quad.point.y += y;
        }
      }
      // This is important, it determines if the quadrant intersects
      // with the node.  If it does not, it returns false
      // and we no longer visit and sub-quadrants or nodes
      // in our quadrant, if true it descends into it
      return x1 > nx2 || x2 < nx1 || y1 > ny2 || y2 < ny1;
    });
  };
}