如何在d3.js/JavaScript中将零值添加到时间序列中

d3n*_*oob 16 javascript d3.js

这是我之前使用php或(我认为是)不必要的复杂MySQL查询解决的问题,但我突然认为在JavaScript/d3.js中必须有一个更优雅的解决方案.

想象一下,我有一个日期和值的数据集,我想在d3.js中转换成条形图.

date,value
2013-01,53
2013-02,165
2013-03,269
2013-04,344
2013-05,376
2013-06,410
2013-07,421
2013-09,376
2013-10,359
2013-11,392
2013-12,433
2014-01,455
2014-02,478
Run Code Online (Sandbox Code Playgroud)

您会注意到数据中的第8个月(8月)没有条目.假设8月是零值,最终结果是得到的条形图看起来正常,但当然没有第8个月应该是间隙(零).

在此输入图像描述

在此输入图像描述

我有一个脚本和数据jsfiddle供参考.

我考虑过尝试添加一个填充了零的完整数据集,然后对其进行迭代以包含数据中的值,但这似乎过于复杂.我假设有一个优雅的解决方案,我太无知了.

谢谢你的帮助.

编辑#1:回应explunit的回答:

理想情况下,解决方案应该是对数据系列的操纵,而不是仅适用于条形图.这意味着在这个jsfiddle的线图的等价物将在它的中间突然下降.

编辑#2:经过一段时间的游戏后:

这里玩了Google网上论坛页面上的建议后,我设法获得了一段代码来完成我想要的工作.它采用时间戳数据,根据时间范围创建一个域,并创建一个单独的数组(在这种情况下).然后我粗略地遍历两个数组,并将适合初始(未完全填充时间值)数组的值添加到具有所有时间值的数组中(并且数据值最初设置为零).

最终结果是一个线图,最初看起来像这样,因为它在2013年7月到9月之间迭代;

在此输入图像描述

随后被渲染为,因为August值将被添加为零;

在此输入图像描述

这里有一个代码jsfiddle ;

我会先说出来的.虽然它在这个实例中完成了我想要的工作,但它距离优雅或可扩展还有很长的路要走.如果比我聪明的人能够看到如何减少攻击性,我将不胜感激.

exp*_*nit 10

我的整体方法没有太大改进,但是如果你使用更多的内置方法并添加下划线/ lodash,你可以使数据转换更短:

x.domain(d3.extent(data, function(d) { return d.date; })).ticks(d3.time.month);
y.domain([0, d3.max(data, function(d) { return d.value; })]);

var newData = x.ticks().map(function(monthBucket) {
    return _.find(data, {date: monthBucket}) || {date: monthBucket, value: 0};
});
Run Code Online (Sandbox Code Playgroud)

如果我们告诉它应该使用每月滴答,那么我们可以再次将ticks数组退出,而不是构建一个单独的桶数组.

然后从那一点开始我们只使用.map而不是for循环和lodash(或下划线)_.find方法来匹配我们的原始数据.更新了小提琴:http://jsfiddle.net/a5jUz/3/


以下原始答案...如果您想使用D3刻度来展开条形图上的值:

1 - 你必须使用时间尺度而不是顺序尺度:

var x = d3.time.scale().range([0, width]);
Run Code Online (Sandbox Code Playgroud)

2 - 您需要根据日期范围的最小值/最大值设置该比例的域:

x.domain(d3.extent(data, function(d) { return d.date; })).nice();
Run Code Online (Sandbox Code Playgroud)

3 - [丑陋的部分]现在您没有使用序数比例,您没有rangeBand条形定位的功能:

  // TODO: calculate based on overall width & number of data points  
  .attr("x", function(d) { return x(d.date); })
  .attr("width", 16)
Run Code Online (Sandbox Code Playgroud)

更新了小提琴:http: //jsfiddle.net/LWyjf/


小智 5

这是不使用lodash / underscore填充零的另一种选择,方法是使用与d3.get()相对的_.find()。虽然不确定这如何影响性能。

var date_range = d3.time.hours(startDate, endDate, 1);

var m = d3.map(data, function(d) { return d.date });
var newData = date_range.map(function(bucket) {
    return m.get(bucket) || {date: bucket, value: 0};
});
Run Code Online (Sandbox Code Playgroud)