“比例尺是一组把输入或映射为输出范围的函数”。-----Mike Bostock

一般而言,任意数据集中的值不可能刚好与图表中的像素尺度一一对应。而D3中,比例尺要做的就是将数据值映射为可视图形中的可替代值得手段。

D3中,比例尺是一种函数,带参数。你可以定义任意多个比例尺函数。

本节中,我们将讨论线性比例尺。当然,还有序数、对数、平方根比例尺等等,但这里我们不做讨论,大家可以以线性比例尺为参考,以此类推。

(1)概念

首先,我们先定义一个数据集:

let dataset = [100,200,300,400,500];

在使用比例尺之前,我们需要理解两个概念:

  • 输入值域:指可能的输入值的范围。即最大值与最小值范围。例如100-500;
  • 输出范围:指输出可能的范围,一般以用于显示的像素为单位;

我们用一幅图来表示:

例如输入值域为[100,500],输出范围[10,350]

比例尺添加pythoncsdn_比例尺添加pythoncsdn

接着,我们来创建比例尺:

D3有一个比例尺生成函数 d3.scale。
例如,let scale = d3.scale.linear();

接着,设定输入值域与输出范围:

let scale = d3.scale.linear().domain([100,500]).range([10,350]);

传参:

scale(100);//返回10

一般而言,我们会在attr()或其他类似方法中调用比例尺函数,而不会像这样独立调用它;下面我们将它应用到散点图中。

(2)应用

你很可能不想给值域设定固定的值,通过d3.min()和d3.max()能帮助你。

例如:

d3.max(dataset,function(d){
    return d[0]; //返嵌套数组中第一个,最大的一个值
});

下面是完整示例:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
    <style>
        div.bar {
            display: inline-block;
            width: 20px;
            height: 75px;
            margin-right: 2px;
            background-color: teal;

        }
    </style>
</head>
<body>

    <script src="https://cdn.bootcss.com/jquery/3.2.1/jquery.js"></script>
    <script src="https://d3js.org/d3.v3.js"></script>
    <script>
        //D3.js code
       let w = 600;
       let h = 100;
       let padding = 20;
       let svg = d3.select("body").append("svg").attr("width",w).attr("height",h);//把append()返回的新元素保存在了变量svg中
        let dataset = [
            [5,20],[480,90],[250,50],[100,33],[330,95],[410,12],[475,44],[25,67],[85,21],[220,88]
            ];
       let xScale = d3.scale.linear()
                            .domain([0,d3.max(dataset,function(d){return d[0];})])
                            .range([padding,w-padding*2])
                            .nice();//nice()告诉比例尺取得为range()设置的任何值域,把两端的值扩展到最接近的整数。如[0.2000011166,0.99999943]优化为[0.2,1]
        let yScale = d3.scale.linear()
                             .domain([0,d3.max(dataset,function(d){return d[1];})])
                             .range([h-padding,padding])
                             .nice();
        let rScale = d3.scale.linear()
                             .domain([0,d3.max(dataset,function(d){return d[1];})])
                             .range([2,5])
                             .nice();              
        svg.selectAll("circle")
           .data(dataset)
           .enter()
           .append("circle")
           .attr("cx",function(d,i){
                return xScale(d[0]); //返回缩放后的值
           })
           .attr("cy",function(d){
               return yScale(d[1]);
           })
           .attr("r",function(d){
               return rScale(d[1]);
           });
           //添加标签     
            svg.selectAll("text")
            .data(dataset)
            .enter()
            .append('text')
            .text(function(d){
                return d[0]+ "," + d[1];//设置标签内容
            })
            .attr({
                fill : "black",
                x : function(d) {return xScale(d[0])+10;},//将标签与散点位置一一对应
                y : function(d) {return yScale(d[1]);}
            })
            .style("font-size", "11px");
    

    </script>
</body>
</html>

比例尺添加pythoncsdn_d3.js_02

(3)其他比例尺

  • sqrt 平方根比例尺;
  • pow 幂比例尺,适合值以指数级变化的数据集;
  • log 对数比例尺;
  • quantize 输出范围为独立的值得线性比例尺,适合想把数据分类的情形;
  • quantile 适合已经对数据分类的情形;
  • ordinal 使用非定量值(如类名)作为输出的序数比例尺;
  • d3.time.scale() 针对日期和时间值得一个比例尺方法,可以对日期刻度作特殊处理;
  • category10\category20\category20b\category20c 能够输出10-20种类别颜色的预设序数比例尺;

(4)下节预告

本节内容就到这里,但是你一定还是觉得缺了些什么。下一节中,我们会给散点图添加上数轴坐标,更形象地从二维视角表现散点图。