【D3.js in Action 3 精译_034】4.1 D3 中的坐标轴的创建(中篇):定义横纵坐标轴的比例尺

当前内容所在位置(可进入专栏查看其他译好的章节内容)

  • 第一部分 D3.js 基础知识
    • 第一章 D3.js 简介(已完结)
      • 1.1 何为 D3.js?
      • 1.2 D3 生态系统——入门须知
      • 1.3 数据可视化最佳实践(上)
      • 1.3 数据可视化最佳实践(下)
      • 1.4 本章小结
    • 第二章 DOM 的操作方法(已完结)
      • 2.1 第一个 D3 可视化图表
      • 2.2 环境准备
      • 2.3 用 D3 选中页面元素
      • 2.4 向选择集添加元素
      • 2.5 用 D3 设置与修改元素属性
      • 2.6 用 D3 设置与修改元素样式
      • 2.7 本章小结
    • 第三章 数据的处理(已完结)
      • 3.1 理解数据
      • 3.2 准备数据
      • 3.3 将数据绑定到 DOM 元素
        • 3.3.1 利用数据给 DOM 属性动态赋值
      • 3.4 让数据适应屏幕
        • 3.4.1 比例尺简介(上篇)
        • 3.4.2 线性比例尺(中篇)
          • 3.4.2.1 基于 Mocha 测试 D3 线性比例尺(DIY 实战)
        • 3.4.3 分段比例尺(下篇)
          • 3.4.3.1 使用 Observable 在线绘制 D3 条形图(DIY 实战)
      • 3.5 加注图表标签(上篇)
        • 3.5.1 人物专访:Krisztina Szűcs(下篇)
      • 3.6 本章小结
    • 第四章 直线、曲线与弧线的绘制 ✔️
      • 4.1 坐标轴的创建(上篇)
        • 4.1.1 D3 中的边距约定(中篇) ✔️
        • 4.1.2 坐标轴的生成(中篇) ✔️
          • 4.1.2.1 比例尺的声明(中篇) ✔️
          • 4.1.2.2 坐标轴的添加(下篇)
          • 4.1.2.3 轴标签的添加(下篇)
      • 4.2 D3 折线图的绘制

文章目录

    • 4.1.1 D3 的边距约定 The margin convention
    • 4.1.2 坐标轴的生成 Generating axes
      • 1 比例尺的声明 Declaring the scales

《D3.js in Action》全新第三版封面

《D3.js in Action》全新第三版封面

译者按
继第四章概述部分交代本章任务,并完成绘制图表前的相关数据准备工作后,本篇开始将逐步深入介绍 D3 的边距约定以及坐标轴的基础核心概念,一定要用心学习。D3 的知识结构非常严密,一步跟不上,后面再想补回来就非常吃力了,主打一个步步为营。不要在意内容的多寡,潜心积累,持续深耕就行了。一起学起来吧!

4.1.1 D3 的边距约定 The margin convention

D3 的边距约定,旨在通过系统化的、可重用的方式为图表周围的坐标轴、标签以及图例保留足够的空间。该约定涉及四个方向,即图表的上方、右侧、下方和左侧,如图 4.3 所示。通过声明这些边距值,就可以知道图表核心区域的位置信息和尺寸大小,该核心区域也被称为 内部图表(inner chart

图 4.3 用 D3 的边距约定确定出的图表上、下、左、右方向上的外边距大小示意图

【图 4.3 用 D3 的边距约定确定出的图表上、下、左、右方向上的外边距大小示意图】

各边距(margin)的值都在一个边距对象中声明。该对象由上、下、左、右边距组成。下面来给我们的折线图创建边距对象。在函数 drawLineChart() 内,声明一个名为 margin 的常量,并沿顺时针方向令其顶部、右侧、底部、左侧的边距值分别为 40px170px25px40px,如以下代码所示:

const drawLineChart = (data) => {
  const margin = {top: 40, right: 170, bottom: 25, left: 40};
};

提前准确预判该为坐标轴和图表标签预留多大空间通常是不现实的。一般得从一个合理的推测开始,然后再根据需要进行调整。例如,查看前面图 4.1 中的折线图或者浏览项目的线上版本(详见 http://mng.bz/5orB),就会发现图表右侧渲染出的标签相对较宽,因此右边距暂定 170px;此外,坐标轴的标签占用空间其实并不大,因此剩余的边距可以调小一点。

一旦确定了边距对象的尺寸,下一步就可以考虑 SVG 容器的尺寸大小了。当 SVG 的容器大小和边距都确定后,就可以算出另两个新常量的大小:innerWidthinnerHeight,分别表示内部图表的宽度和高度。各尺寸的空间分布情况如图 4.4 所示:

图 4.4 SVG 容器尺寸和边距确定后,就能算出内部图表的宽高

【图 4.4 SVG 容器尺寸和边距确定后,就能算出内部图表的宽高】

内部图表的宽度值等于 SVG 容器的宽度减左右两侧的边距宽度。若 SVG 容器宽 1000px,左右边距分别为 170px40px,则内部图表可分到 790px;同理,若 SVG 容器总高度为 500px,减去上下边距的高即为内部图表的高度值,可算得 435px。这样,常量 innerWidthinnerHeight 就与边距尺寸成线性关系,后续如果边距有变动,这两个值也会同步更新:

const margin = {top: 40, right: 170, bottom: 25, left: 40};
const width = 1000;
const height = 500;
const innerWidth = width - margin.left - margin.right;
const innerHeight = height - margin.top - margin.bottom;

接着再来添加折线图的 SVG 容器。还是在函数 drawLineChart() 内部,将一个 SVG 元素追加到 idline-chartdiv 元素内;然后使用 widthheight 常量设置 SVG 的 viewBox 属性。还可以临时给 SVG 元素绘制一个边框,以便看清工作区域的方位(关于如何将元素添加到 DOM 结构、或者元素属性与样式的设置方法,详见第 2 章相关内容)。

const svg = d3.select("#line-chart")
  .append("svg")
  .attr("viewBox", `0, 0, ${width}, ${height}`);

现在,各方向上的边距值和 SVG 的原点坐标都已经确定好了。折线图中的每个元素都必须向新划定的绘图区域平移。这时应该将内部图表封装在一个 SVG 的分组元素中,并只对该分组元素设置平移即可,无需挨个为每个图表元素指定平移量,如图 4.5 所示。这样做也为内部图表建立了一套新的坐标系。

图 4.5 包含内部图表的 SVG 分组元素的平移效果图。该平移也为内部图表各元素建立了新的坐标系

【图 4.5 包含内部图表的 SVG 分组元素的平移效果图。该平移也为内部图表各元素建立了新的坐标系】

要实现上述效果,需要在 SVG 容器中添加一个分组元素;然后利用左边距和上边距指定该分组的平移量;最后将 SVG 分组赋给常量 innerChart,以备后用:

const innerChart = svg
  .append("g")
  .attr("transform", `translate(${margin.left}, ${margin.top})`);

D3 的边距约定以及上述做法的主要优点在于,这些尺寸一旦确立,后续就无需再考虑相关问题了。这样,在充分考虑并提前预留标签、图例及其他补充信息的绘制区域后,就可以继续创建折线图的坐标轴以及核心图表了。

4.1.2 坐标轴的生成 Generating axes

边距一旦确定,就可以给图表添加坐标轴了。坐标轴既是数据可视化的重要组成部分,同时也是观众理解所绘数据与相关类别的重要参考。

对照前面图 4.1 或者线上版(详见:http://mng.bz/5orB)的折线图,可以看到两个坐标轴:横轴(也称 x 轴)显示各月份;纵轴(或 y 轴)则为华氏温度 1 的参考轴线。

译注

为方便查看效果,这里直接给出本专栏第 32 篇(第四章概述)中的最终效果图:

补图 4.1 本章实现项目:2021 年纽约市温度变化及全年降水天数占比情况可视化

【补图 4.1 本章实现项目:2021 年纽约市温度变化及全年降水天数占比情况可视化】

D3 通过 axis()组件生成器(component generator) 来创建坐标轴。该生成器接受一个比例尺函数作为参数,并返回一个组成该坐标轴的 SVG 元素。第三章介绍的比例尺相关知识还有印象吗,它们负责将定义域中的值转换为值域中对应的值,并通过这些值来绘制图表。而折线图与之类似,比例尺传入后,将算得数据集各日期的水平坐标,以及相关温度值对应的垂直坐标。

1 比例尺的声明 Declaring the scales

根据上面的介绍,创建 D3 坐标轴的第一步,其实是声明它们相应的比例尺。首先要创建一个能设置日期水平位置的比例尺,这正是 D3 时间比例尺 d3.scaleTime() 的长项(更多 D3 比例尺选型方面的介绍,详见本书 附录 B(译注:将择日翻译))。时间比例尺属于第三章介绍过的四类比例尺中的第一类:接受连续型输入,并返回连续型输出。时间比例尺的行为特征与第三章介绍的线性比例尺非常类似,唯一的区别在于这里处理的是与时间相关的数据。

接下来,先声明一个时间比例尺常量 xScale,负责 x 轴上的元素定位。该比例尺的定义域从第一个日期值开始,到最后一个日期结束。以下代码片段还用到了 d3.min()d3.max() 函数,它们是 d3-array 模块中的两个工具方法,专门用于查找对应的极值。

而时间比例尺的值域,则是在内部图表的可用水平空间上均匀分布(如图 4.5 所示)。在内部图表的坐标系中,意味着值域将从 0 延展至之前计算过的内部宽度 innerWidth(比例尺定义域及值域的相关知识点,详见第三章):

const firstDate = d3.min(data, d => d.date);
const lastDate = d3.max(data, d => d.date);
const xScale = d3.scaleTime()
  .domain([firstDate, lastDate])
  .range([0, innerWidth]);

而温度则沿 y 轴分布,也需要一个连续输入与输出的比例尺。这里自然首选线性比例尺,因为我们希望温度与折线图上的垂直坐标呈线性比例关系。

以下代码片段中,声明了一个表示温度比例尺的常量 yScale,负责 y 轴上的元素定位。假定 y 轴从 0 开始,因此 0 即为定义域的第一个传入值;虽然数据集中的最低温度大约在 26°F 左右(即 -3.3°C 上下),但将 y 轴设为从 0 开始也不错,况且在本例中也可以准确地看出气温的演变情况。这就像生活中的大多数情况,其实并没有一个硬性规定;同理,本例中的图表也不能因为华氏度中的 0 度并非真正的 0 而有对错之分。

而定义域需要的第二个参数,这里选数据集中的最大温度。具体的值通过使用 d3.max() 函数检索数据集的 max_temp_F 字段来确定。

比例尺的值域沿图表高度均匀分布,但由于 SVG 坐标系的垂直坐标是向下为正的,因此值域即从图表的内部高度 innerHeight 开始,也就是图表的左下角位置;终点则为 0,对应其左上角的纵坐标:

const maxTemp = d3.max(data, d => d.max_temp_F);
const yScale = d3.scaleLinear()
  .domain([0, maxTemp])
  .range([innerHeight, 0]);

译注
由于 4.1.2 节篇幅太长,拟再分三个子篇进行介绍(足见 D3 坐标轴这个知识点的信息密度之大)。本章同步源码已上传 CSDN 资源库。章节内容持续更新中,敬请关注本精译专栏,及时获取 D3.js 最前沿的数据可视化主流趋势及最佳实践。



  1. 华氏温度,即 Fahrenheit temperature,是由德国物理学家丹尼尔·华氏(Daniel Gabriel Fahrenheit)于 1701 年提出,其刻度是基于他所使用的水银温度计的特点,并将其划分为180个单位,并在标准大气压下测得水的冰点为华氏 32 度(32°F),沸点为华氏 212 度(212°F)。因为历史文化等原因,美国是目前世界上仅有的五个还在使用华氏度的国家之一。华氏度与摄氏度的换算关系为:1°F = 1.8 × °C + 32。 ↩︎

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/895331.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

企业资产安全之数据防泄密要领

在数字化时代,数据已成为企业最宝贵的资产之一。然而,随着数据价值的增加,数据泄露的风险也随之上升。从内部员工的无意泄露到外部黑客的恶意攻击,企业数据安全面临着前所未有的挑战。SDC沙盒数据防泄密解决方案,正是为…

用 Python 构建高级配对交易策略

作者:老余捞鱼 原创不易,转载请标明出处及原作者。 写在前面的话: 本文阐述通过分析加密货币和传统金融工具之间的相关性和协整性,以及实施 Z-score 方法来生成交易信号,然后介绍如何使用 Python 构建配对交易策…

无人机搭载激光雷达在地形测绘中的多元应用

一、高精度地形测量 无人机激光雷达能够发射激光脉冲并接收其回波,通过精确计算激光脉冲的往返时间来确定目标物的距离。这一特性使得无人机激光雷达在地形测绘中能够实现高精度的三维地形测量。通过快速获取大量地形数据,可以生成高精度的数字高程模型…

VScode背景更改

效果 实现方法 第0步 以管理员身份运行VScode 首先 需要安装这个扩展 然后 接下来 找到配置文件 再后来 在配置文件的下面但不超过最后一个大括号的地方加入以下内容 "update.enableWindowsBackgroundUpdates": true,"background.fullscreen": {…

Gee引擎配置微端后登录游戏黑屏怎么办?

GEE引擎配置微端后登录游戏黑屏怎么办?今天飞飞和你们分享GEE引擎配置微端后游戏黑屏的解决办法,希望可以帮助到你~ 1、端口不对 微端没有更新,玩家进入游戏是地图跟装备都看不见,是漆黑的,微端显示连接失败&#xff…

Leecode刷题之路第26天之删除有序数组中的重复项

题目出处 26-删除有序数组中的重复项-题目出处 题目描述 给你一个 非严格递增排列 的数组 nums ,请你 原地 删除重复出现的元素,使每个元素 只出现一次 ,返回删除后数组的新长度。元素的 相对顺序 应该保持 一致 。然后返回 nums 中唯一元…

鸿蒙网络编程系列31-使用RCP调用OpenAI接口实现智能助手

简介 在OpenAI推出GPT系列大模型以后,市场上各种类似的大模型也层出不穷,这些大模型也基本都会兼容OpenAI的接口,在开发基于大模型的应用时,选择使用OpenAI接口作为和后端大模型通讯的标准,可以更好的适配不同厂家的模…

2024年五一杯数学建模C题煤矿深部开采冲击地压危险预测求解全过程论文及程序

2024年五一杯数学建模 C题 煤矿深部开采冲击地压危险预测 原题再现: “煤炭是中国的主要能源和重要的工业原料。然而,随着开采深度的增加,地应力增大,井下煤岩动力灾害风险越来越大,严重影响着煤矿的安全高效开采。在…

一个人如何开发一款App软件

个人开发软件和公司开发软件不一样,其中就是收费上,个人开发的费用低,售后服务态度好啊。一个人负责开发也负责售后,客户就你一个。一般都是工作室和个人接单的多,不是太大的项目就建议是个人开发吧,因为能…

网络编程(21)——通过beast库快速实现http服务器

目录 二十一、day21 1. 头文件和作用域重命名 2. reponse时调用的一些函数 3. http_connection a. 构造函数 b. start() c. process_request() d. create_response() e. create_post_response() f. write_response() 4. Server 5. 主函数 6. 测试 1)测…

零基础入门人工智能,如何利用AI工具提升你的学习效率?

在这个信息爆炸的时代,人工智能(AI)不仅是技术行业的热词,更是我们日常生活中不可或缺的部分。你是否也想过,如何更有效地学习和利用这些强大的AI工具来提升自己的学习效率?今天,我们将介绍六款…

electron本地OCR实现

使用tesseract.js - npm (npmjs.com) 官方demo&#xff1a;GitHub - Balearica/tesseract.js-electron: An example to use tesseract.js in electron 目录结构&#xff1a; // 引入 <script type"module" src"./ocr/tesseract.js"></script>…

垃圾收集器与内存分配机制(三)

目录 学习前言 一、低延迟垃圾收集器 1. Shenandoah收集器 二、ZGC 1. 内存布局 2. 更巧妙的并发整理 三、其他垃圾收集器 学习前言 除了之前我们所学习的经典垃圾收集器除外&#xff0c;我们还有一些低延迟垃圾收集等&#xff01; 之前所学经典垃圾收集器&#xff0c;…

ubuntu 安装nginx

sudo apt-get update sudo apt-get install nginx sudo nginx -vsudo systemctl status nginx sudo systemctl start nginx sudo systemctl stop nginx sudo systemctl restart nginx#浏览器输入&#xff1a;http://192.168.31.181/#查看文件结构 cd /etc/nginx sudo cp nginx.…

瑞云快图云渲染怎么样?渲染一张图贵吗?

在如今的数字时代&#xff0c;云渲染已经成为了设计师和建筑师们不可或缺的工具。而瑞云快图作为一款备受瞩目的云渲染平台&#xff0c;以其出色的性能和实惠的价格吸引了众多用户。 那么&#xff0c;瑞云快图云渲染究竟怎么样&#xff1f;渲染一张图贵吗&#xff1f;本文将为…

kernel32.dll下载地址:如何安全地恢复系统文件

关于从网络上寻找kernel32.dll的下载地址&#xff0c;这通常不是一个安全的做法&#xff0c;而且可能涉及到多种风险。kernel32.dll是Windows操作系统的核心组件之一&#xff0c;负责内存管理、进程和线程管理以及其他关键系统功能。因为kernel32.dll是系统的基础文件&#xff…

初试PostgreSQL数据库

文章目录 一、PostgreSQL数据库概述1.1 PostgreSQL的历史1.2 PostgreSQL安装1.3 安装PostgreSQL二、PostgreSQL起步2.1 连接数据库2.1.1 SQL Shell2.1.2 执行SQL语句2.2 pgAdmin 42.2.1 打开pgAdmin 42.2.2 查找数据库2.2.3 打开查询工具2.2.4 执行SQL语句三、实战小结文章目录…

使用cmdline-tools安装Android SDK与NDK

1.下载SDK工具: www.android.com 选择下载平台包 同意并下载Command Line Tools 下载中 下载完成后解压 2. 创建android sdk目录并复制sdk工具 创建目录

一文彻底弄懂MySQL的MVCC多版本控制器

InnoDB 的 MVCC&#xff08;Multi-Version Concurrency Control&#xff0c;多版本并发控制&#xff09; 是 MySQL 实现高并发事务处理的一种机制。通过 MVCC&#xff0c;InnoDB 可以在高并发环境下支持 事务隔离&#xff0c;并提供 非阻塞的读操作&#xff0c;从而避免锁定所有…

Docker配置网站环境

Mysql 先安装mysql 启动并后台运行&#xff1a;run -d 容器名称&#xff1a;--name 设置端口映射&#xff1a;-p 主机端口&#xff1a;容器端口 环境变量&#xff1a;-e 最后指定镜像名称 sudo docker run -d \--name mysql\-p 3306:3306\-e MYSQL_ROOT_PASSWORD123456\…