在D3.js中绘制多条线

win*_*ter 39 line d3.js

到目前为止,我一直在使用循环来为D3可视化添加线元素,但这似乎并不符合API的精神.

假设我有一些数据,

var data = {time: 1, value: 2, value2: 5, value3: 3,value4: 2},
           {time: 2, value: 4, value2: 9, value3: 2,value4: 4},
           {time: 3, value: 8, value2:12, value3: 2,value4:15}]);
Run Code Online (Sandbox Code Playgroud)

我想要四行,时间作为所有4的X.

我可以这样做:

var l = d3.svg.line()
        .x(function(d){return xScale(d[keys[0]]);})
        .y(function(d,i){
            return yScale(d[keys[1]]);})
        .interpolate("basis");
var l2 = d3.svg.line()
        .x(function(d){return xScale(d[keys[0]]);})
        .y(function(d,i){
            return yScale(d[keys[2]]);})
        .interpolate("basis");
var l3 = d3.svg.line()
        .x(function(d){return xScale(d[keys[0]]);})
        .y(function(d,i){
            return yScale(d[keys[3]]);})
        .interpolate("basis");  
var l4 = d3.svg.line()
        .x(function(d){return xScale(d[keys[0]]);})
        .y(function(d,i){
            return yScale(d[keys[4]]);})
        .interpolate("basis");
Run Code Online (Sandbox Code Playgroud)

然后逐个添加(或循环).

var line1 = group.selectAll("path.path1")
        .attr("d",l(data));
var line2 = group.selectAll("path.path2")
        .attr("d",l2(data));
var line3 = group.selectAll("path.path3")
        .attr("d",l3(data));
var line4 = group.selectAll("path.path4")
        .attr("d",l4(data));
Run Code Online (Sandbox Code Playgroud)

是否有更好的更通用的方法来添加这些路径?

mbo*_*ock 74

是.首先,我将重构您的数据以便更容易迭代,如下所示:

var series = [
  [{time: 1, value: 2}, {time: 2, value: 4}, {time: 3, value: 8}],
  [{time: 1, value: 5}, {time: 2, value: 9}, {time: 3, value: 12}],
  [{time: 1, value: 3}, {time: 2, value: 2}, {time: 3, value: 2}],
  [{time: 1, value: 2}, {time: 2, value: 4}, {time: 3, value: 15}]
];
Run Code Online (Sandbox Code Playgroud)

现在您只需要一个通用线:

var line = d3.svg.line()
    .interpolate("basis")
    .x(function(d) { return x(d.time); })
    .y(function(d) { return y(d.value); });
Run Code Online (Sandbox Code Playgroud)

然后,您可以一次性添加所有路径元素:

group.selectAll(".line")
    .data(series)
  .enter().append("path")
    .attr("class", "line")
    .attr("d", line);
Run Code Online (Sandbox Code Playgroud)

如果要使数据结构格式更小,还可以将时间提取到单独的数组中,然后使用2D数组作为值.这看起来像这样:

var times = [1, 2, 3];

var values = [
  [2, 4, 8],
  [5, 9, 12],
  [3, 2, 2],
  [2, 4, 15]
];
Run Code Online (Sandbox Code Playgroud)

由于矩阵不包含时间值,因此需要从线生成器的x-accessor中查找.另一方面,y-accessor被简化,因为您可以将矩阵值直接传递给y-scale:

var line = d3.svg.line()
    .interpolate("basis")
    .x(function(d, i) { return x(times[i]); })
    .y(y);
Run Code Online (Sandbox Code Playgroud)

创建元素保持不变:

group.selectAll(".line")
    .data(values)
  .enter().append("path")
    .attr("class", "line")
    .attr("d", line);
Run Code Online (Sandbox Code Playgroud)

  • `x`和`y`指的是尺度,你应该在使用之前定义. (15认同)
  • 这完整吗?`group.selectAll`来自何处,我没有看到组对象的定义. (4认同)