转载请注明出处!
比例尺简述:
比例尺是一组把输入域映射到输出范围的函数。
一般来说数据集中的值不可能恰好与图表中的像素尺度一一对应。比例尺就是把这些数据值映射到可视化图形中使用的新值的便捷手段。
D3的比例尺就是那些你定义的带有参数的函数。
听到比例尺有些人就会想到最终图表中一系列的刻度线,对应一系列的值,不要搞错,这些刻度显示坐标轴的一部分,而坐标轴只是比例尺的一种形象的表示。比例尺实际上代表的是一种数学关系,不可能直接输出可见的图形。比例尺和坐标轴是两种不同但相关的东西。
下面我们只讨论线性比例尺。
值域和范围:
比例尺的输入值域(input domain)指的是可能的输入值的范围。
比例尺的输出范围(output range)指的是输出值的可能范围。
其实就是归一化,对于线性比例尺,d3可以帮助我们处理归一化过程的数学计算:输入值根据值域先进行归一化,然后再把归一化后的值对应到输出范围。
创建比例尺:
var scale = d3.scale.linear() .domain([100, 500]) .range([10, 350]);
例如:
scale(100)输出是10,scale(300)输出是180…
缩放散点图:
var dataset = [
[5, 20], [480, 90], [250, 50], [100, 33], [330, 95],
[410, 12], [475, 44], [25, 67], [85, 21], [220, 88]
];
返回所有坐标中X值中最大的:
d3.max(dataset, function(d) { //返回 480 return d[0]; //每一个子数组中的第一个位置的值 });
X轴缩放:
var xScale = d3.scale.linear() .domain([0, d3.max(dataset, function(d) { return d[0]; })]) .range([0, w]);
Y轴缩放:
var yScale = d3.scale.linear() .domain([0, d3.max(dataset, function(d) { return d[1]; })]) .range([0, h]);
设定圆心的坐标(注意使用和坐标同样缩放尺度的坐标值):
.attr("cx", function(d) {
return d[0];
})
缩放后的坐标X值:
.attr("cx", function(d) { return xScale(d[0]); })
Y值同样如此:
.attr("cy", function(d) {
return d[1];
})
缩放后的坐标Y值:
.attr("cy", function(d) { return yScale(d[1]); })
设定文本坐标值(同上):
.attr("x", function(d) {
return d[0];
})
.attr("y", function(d) {
return d[1];
})
变成:
.attr("x", function(d) { return xScale(d[0]); }) .attr("y", function(d) { return yScale(d[1]); })
源代码:
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title>testD3-10-scale.html</title> <script type="text/javascript" src="d3.v3.js"></script> <style type="text/css"> </style> </head> <body> <script type="text/javascript"> //Width and height var w = 500; var h = 100; var dataset = [ [5, 20], [480, 90], [250, 50], [100, 33], [330, 95], [410, 12], [475, 44], [25, 67], [85, 21], [220, 88] ]; //Create scale functions var xScale = d3.scale.linear() .domain([0, d3.max(dataset, function(d) { return d[0]; })]) .range([0, w]); var yScale = d3.scale.linear() .domain([0, d3.max(dataset, function(d) { return d[1]; })]) .range([0, h]); //Create SVG element var svg = d3.select("body") .append("svg") .attr("width", w) .attr("height", h); svg.selectAll("circle") .data(dataset) .enter() .append("circle") .attr("cx", function(d) { return xScale(d[0]); }) .attr("cy", function(d) { return yScale(d[1]); }) .attr("r", function(d) { return Math.sqrt(h - d[1]); }); svg.selectAll("text") .data(dataset) .enter() .append("text") .text(function(d) { return d[0] + "," + d[1]; }) .attr("x", function(d) { return xScale(d[0]); }) .attr("y", function(d) { return yScale(d[1]); }) .attr("font-family", "sans-serif") .attr("font-size", "11px") .attr("fill", "red"); </script> </body> </html>
修饰图表:
点大小与圈大小成正比,想把大的放在下面,只要改变Y轴值域倒转即可: .range([h , 0]);
有些圆形的边缘会被切掉一部分,为此引入一个边距变量:
var padding = 20;
以便在设置两个比例尺的时候加入边距,边距可以把圆形往里推,使他们远离SVG的四边,从而避免被切掉。
xScale的返回改为:.range([padding, w-padding]);
但右边的文本仍然会被切掉,可以增大边距2*padding。
原来我们是把每个圆形的半径设置为y值的平方根,现在,我们同样可以为半径设置一个比例尺:
var rScale = d3.scale.linear() .domain([0, d3.max(dataset,function(d) {
return d[1];
})])
.rangeRound([2, 5]);
然后再这样设定圆的半径:
.attr("r", function(d) {
return rScale(d[1]);
});
这样所有远的半径就永远介于2~5之间。
其他方法:
nice():
告诉比例尺取得range()设置的任何值域,把两端的值扩展到最接近的整数。“For example, for a domain of [0.20147987687960267, 0.996679553296417], the nice domain is [0.2, 1].”
rangeRound():
代替range之后,比例尺输出的所有值都会舍入到最接近的整数值。对输出值取整有助于图形对应精确地像素值,避免边缘出现模糊不清的锯齿。
clamp():
默认情况下,比例尺可以返回指定范围之外的值。比如,假设给定的值位于输入值域之外,那么比例尺也会返回一个位于输出范围之外的值。调用clamp之后就会强行所有输出值都要在指定范围内。意味着超出范围,会取整到范围的最大值或最小值。
源代码微调:
<!DOCTYPE html> <html lang="en"> <head> <title>D3: Linear scales with a scatterplot</title> <script type="text/javascript" src="../d3/d3.js"></script> <style type="text/css"> /* No style rules here yet */ </style> </head> <body> <script type="text/javascript"> var w = 500; var h = 300; var padding = 20; var dataset = [ [5, 20], [480, 90], [250, 50], [100, 33], [330, 95], [410, 12], [475, 44], [25, 67], [85, 21], [220, 88], [600, 150] ]; //create scales function var xScale = d3.scale.linear() .domain([0, d3.max(dataset,function (d) {return d[0];})]) .rangeRound([padding,w-padding*2]); var yScale = d3.scale.linear() .domain([0, d3.max(dataset,function (d) {return d[1];})]) .rangeRound([h-padding,padding]); var rScale = d3.scale.linear() .domain([0, d3.max(dataset,function (d) {return d[1];})]) .rangeRound([2, 5]); //create SVG element var svg = d3.select("body") .append("svg") .attr("width", w) .attr("height", h); //create circles svg.selectAll("circle") .data(dataset) .enter() .append("circle") .attr("cx", function(d) { return xScale(d[0]); }) .attr("cy", function(d) { return yScale(d[1]); }) .attr("r", function(d) { return rScale(d[1]); }); //create text elements svg.selectAll("text") .data(dataset) .enter() .append("text") .text(function(d){ return d[0] + "," + d[1]; }) .attr("x", function(d) { return xScale(d[0]); }) .attr("y", function(d) { return yScale(d[1]); }) .attr("font-family", "sans-serif") .attr("font-size", "11px") .attr("fill", "red"); </script> </body> </html>
效果图:
For my lover, CC!
参考书籍:《数据可视化实战》