当前位置: 首页 > article >正文

数据可视化:使用D3.js创建交互式图表

数据可视化使用D3.js创建交互式图表大家好我是欧阳瑞Rich Own。今天想和大家聊聊数据可视化这个话题。作为一个全栈开发者我经常需要将复杂的数据以直观的方式展示给用户。D3.js是一个功能强大的数据可视化库今天就来分享一下如何使用D3.js创建交互式图表。为什么选择D3.jsD3.jsData-Driven Documents是一个用于创建动态、交互式数据可视化的JavaScript库。它的优势在于特性说明灵活性完全控制DOM和SVG数据驱动直接绑定数据到DOM丰富的API支持多种图表类型社区活跃大量教程和插件环境准备!-- 在HTML中引入D3.js -- script srchttps://d3js.org/d3.v7.min.js/script或者使用npmnpm install d3基础图表柱状图创建SVG容器// 设置画布尺寸 const width 800; const height 400; const margin { top: 20, right: 30, bottom: 40, left: 50 }; // 创建SVG容器 const svg d3.select(#chart) .append(svg) .attr(width, width) .attr(height, height); // 创建绘图区域 const g svg.append(g) .attr(transform, translate(${margin.left}, ${margin.top})); const innerWidth width - margin.left - margin.right; const innerHeight height - margin.top - margin.bottom;准备数据const data [ { category: A, value: 40 }, { category: B, value: 60 }, { category: C, value: 30 }, { category: D, value: 80 }, { category: E, value: 50 } ];创建比例尺// X轴比例尺序数比例尺 const xScale d3.scaleBand() .domain(data.map(d d.category)) .range([0, innerWidth]) .padding(0.2); // Y轴比例尺线性比例尺 const yScale d3.scaleLinear() .domain([0, d3.max(data, d d.value)]) .range([innerHeight, 0]);绘制坐标轴// X轴 g.append(g) .attr(transform, translate(0, ${innerHeight})) .call(d3.axisBottom(xScale)); // Y轴 g.append(g) .call(d3.axisLeft(yScale));绘制柱状图// 绑定数据并创建矩形 g.selectAll(rect) .data(data) .enter() .append(rect) .attr(x, d xScale(d.category)) .attr(y, d yScale(d.value)) .attr(width, xScale.bandwidth()) .attr(height, d innerHeight - yScale(d.value)) .attr(fill, #00ffff) .attr(opacity, 0.7);交互式图表添加悬停效果g.selectAll(rect) .data(data) .enter() .append(rect) .attr(x, d xScale(d.category)) .attr(y, d yScale(d.value)) .attr(width, xScale.bandwidth()) .attr(height, d innerHeight - yScale(d.value)) .attr(fill, #00ffff) .attr(opacity, 0.7) .on(mouseenter, function(event, d) { d3.select(this) .transition() .duration(200) .attr(fill, #ff00ff) .attr(opacity, 1); }) .on(mouseleave, function(event, d) { d3.select(this) .transition() .duration(200) .attr(fill, #00ffff) .attr(opacity, 0.7); });添加工具提示// 创建工具提示 const tooltip d3.select(#chart) .append(div) .attr(class, tooltip) .style(opacity, 0) .style(position, absolute) .style(background, black) .style(color, #00ffff) .style(padding, 8px) .style(border-radius, 4px); // 绑定工具提示事件 g.selectAll(rect) .on(mouseenter, function(event, d) { tooltip.transition() .duration(200) .style(opacity, 0.9); tooltip.html(类别: ${d.category}br/值: ${d.value}) .style(left, (event.pageX 10) px) .style(top, (event.pageY - 28) px); }) .on(mouseleave, function() { tooltip.transition() .duration(200) .style(opacity, 0); });折线图准备时序数据const timeData [ { date: 2023-01-01, value: 30 }, { date: 2023-01-02, value: 45 }, { date: 2023-01-03, value: 35 }, { date: 2023-01-04, value: 50 }, { date: 2023-01-05, value: 42 }, { date: 2023-01-06, value: 60 }, { date: 2023-01-07, value: 55 } ];创建时间比例尺// 解析日期 const parseDate d3.timeParse(%Y-%m-%d); timeData.forEach(d { d.date parseDate(d.date); }); // X轴时间比例尺 const xTimeScale d3.scaleTime() .domain(d3.extent(timeData, d d.date)) .range([0, innerWidth]); // Y轴线性比例尺 const yTimeScale d3.scaleLinear() .domain([0, d3.max(timeData, d d.value)]) .range([innerHeight, 0]);创建折线生成器const line d3.line() .x(d xTimeScale(d.date)) .y(d yTimeScale(d.value)) .curve(d3.curveMonotoneX); // 平滑曲线绘制折线// 绘制折线 g.append(path) .datum(timeData) .attr(fill, none) .attr(stroke, #ff00ff) .attr(stroke-width, 2) .attr(d, line); // 绘制数据点 g.selectAll(circle) .data(timeData) .enter() .append(circle) .attr(cx, d xTimeScale(d.date)) .attr(cy, d yTimeScale(d.value)) .attr(r, 5) .attr(fill, #00ffff);饼图// 数据 const pieData [ { category: A, value: 30 }, { category: B, value: 20 }, { category: C, value: 50 } ]; // 创建SVG const pieSvg d3.select(#pie-chart) .append(svg) .attr(width, 400) .attr(height, 400); const pieG pieSvg.append(g) .attr(transform, translate(200, 200)); // 创建饼图生成器 const pie d3.pie() .value(d d.value); // 创建弧形生成器 const arc d3.arc() .innerRadius(0) .outerRadius(150); // 颜色比例尺 const color d3.scaleOrdinal() .domain(pieData.map(d d.category)) .range([#00ffff, #ff00ff, #ffff00]); // 绘制饼图 pieG.selectAll(path) .data(pie(pieData)) .enter() .append(path) .attr(d, arc) .attr(fill, d color(d.data.category)) .attr(opacity, 0.8) .on(mouseenter, function(event, d) { d3.select(this) .transition() .duration(200) .attr(opacity, 1); }) .on(mouseleave, function(event, d) { d3.select(this) .transition() .duration(200) .attr(opacity, 0.8); }); // 添加标签 pieG.selectAll(text) .data(pie(pieData)) .enter() .append(text) .attr(transform, d translate(${arc.centroid(d)})) .attr(text-anchor, middle) .attr(fill, white) .text(d d.data.category);力导向图// 创建力导向图 const forceData { nodes: [ { id: A, group: 1 }, { id: B, group: 1 }, { id: C, group: 2 }, { id: D, group: 2 }, { id: E, group: 3 } ], links: [ { source: A, target: B }, { source: B, target: C }, { source: C, target: D }, { source: D, target: E }, { source: E, target: A } ] }; const forceSvg d3.select(#force-chart) .append(svg) .attr(width, 600) .attr(height, 400); const forceSimulation d3.forceSimulation(forceData.nodes) .force(link, d3.forceLink(forceData.links).id(d d.id)) .force(charge, d3.forceManyBody()) .force(center, d3.forceCenter(300, 200)); // 绘制连线 const link forceSvg.append(g) .selectAll(line) .data(forceData.links) .enter() .append(line) .attr(stroke, #00ffff) .attr(stroke-opacity, 0.5); // 绘制节点 const node forceSvg.append(g) .selectAll(circle) .data(forceData.nodes) .enter() .append(circle) .attr(r, 10) .attr(fill, #ff00ff) .call(d3.drag() .on(start, dragstarted) .on(drag, dragged) .on(end, dragended)); // 更新位置 forceSimulation.on(tick, () { link .attr(x1, d d.source.x) .attr(y1, d d.source.y) .attr(x2, d d.target.x) .attr(y2, d d.target.y); node .attr(cx, d d.x) .attr(cy, d d.y); }); function dragstarted(event, d) { if (!event.active) forceSimulation.alphaTarget(0.3).restart(); d.fx d.x; d.fy d.y; } function dragged(event, d) { d.fx event.x; d.fy event.y; } function dragended(event, d) { if (!event.active) forceSimulation.alphaTarget(0); d.fx null; d.fy null; }实战案例加密货币价格图表// 获取加密货币数据 async function fetchCryptoData() { const response await fetch(https://api.coingecko.com/api/v3/coins/markets?vs_currencyusdidsbitcoin,ethereum,solanaordermarket_cap_descper_page3page1sparklinefalse); const data await response.json(); return data; } // 创建加密货币价格图表 async function createCryptoChart() { const data await fetchCryptoData(); const width 600; const height 400; const margin { top: 20, right: 20, bottom: 40, left: 60 }; const svg d3.select(#crypto-chart) .append(svg) .attr(width, width) .attr(height, height); const g svg.append(g) .attr(transform, translate(${margin.left}, ${margin.top})); const innerWidth width - margin.left - margin.right; const innerHeight height - margin.top - margin.bottom; const xScale d3.scaleBand() .domain(data.map(d d.name)) .range([0, innerWidth]) .padding(0.3); const yScale d3.scaleLinear() .domain([0, d3.max(data, d d.current_price)]) .range([innerHeight, 0]); g.append(g) .attr(transform, translate(0, ${innerHeight})) .call(d3.axisBottom(xScale)); g.append(g) .call(d3.axisLeft(yScale).tickFormat(d $ d)); const colors [#f7931a, #627eea, #00ffa3]; g.selectAll(rect) .data(data) .enter() .append(rect) .attr(x, d xScale(d.name)) .attr(y, d yScale(d.current_price)) .attr(width, xScale.bandwidth()) .attr(height, d innerHeight - yScale(d.current_price)) .attr(fill, (d, i) colors[i % colors.length]) .attr(opacity, 0.8) .on(mouseenter, function(event, d) { d3.select(this) .transition() .duration(200) .attr(opacity, 1); }) .on(mouseleave, function(event, d) { d3.select(this) .transition() .duration(200) .attr(opacity, 0.8); }); } createCryptoChart();总结D3.js是一个功能强大的数据可视化工具掌握它可以让你创建出各种精美的交互式图表。从简单的柱状图到复杂的力导向图D3.js都能胜任。我的鬃狮蜥Hash对数据可视化也很感兴趣——它喜欢盯着屏幕上闪烁的图表仿佛在分析数据趋势。也许有一天我会为它创建一个蟋蟀价格指数的实时图表。如果你有数据可视化方面的问题欢迎留言交流我是欧阳瑞极客之路永无止境技术栈D3.js · SVG · JavaScript · 数据可视化

相关文章:

数据可视化:使用D3.js创建交互式图表

数据可视化:使用D3.js创建交互式图表 大家好,我是欧阳瑞(Rich Own)。今天想和大家聊聊数据可视化这个话题。作为一个全栈开发者,我经常需要将复杂的数据以直观的方式展示给用户。D3.js是一个功能强大的数据可视化库&am…...

3分钟解锁WeMod高级功能:开源工具Wand-Enhancer完全指南

3分钟解锁WeMod高级功能:开源工具Wand-Enhancer完全指南 【免费下载链接】Wand-Enhancer Advanced UX and interoperability extension for Wand (WeMod) app 项目地址: https://gitcode.com/gh_mirrors/we/Wand-Enhancer 你是否因为WeMod的高级功能需要付费…...

用STM32F103C8T6驱动Ra-01SC模组:从接线到收发数据的保姆级避坑指南

STM32F103C8T6与Ra-01SC模组实战:从硬件搭建到数据收发的完整解决方案 1. 项目准备与环境搭建 第一次接触LoRa通信时,我拿着两块Ra-01SC模组和STM32开发板,满心期待能快速实现无线数据传输。但现实很快给我上了一课——接线错误导致模组发热、…...

STM32H743 FDCAN实战:手把手教你调试CAN节点错误计数器与Bus_Off状态

STM32H743 FDCAN实战:从寄存器到代码的Bus_Off恢复指南 当你的STM32H743项目突然出现CAN通信中断,调试器里FDCAN_PSR寄存器的BOFF位亮起红灯时,真正的挑战才刚刚开始。这不是普通的通信故障,而是触发了CAN协议中最严厉的惩罚机制—…...

CC2530开发避坑指南:IAR for 8051 10.10.1新建工程到流水灯调试的完整流程

CC2530开发实战:IAR for 8051 10.10.1工程搭建与调试全解析 第一次接触CC2530和IAR开发环境时,我盯着满屏的编译错误和无法识别的仿真器,深刻理解了什么叫"从入门到放弃"。这种经历在嵌入式开发领域太常见了——特别是当你面对的是…...

Linux重定向与管道:从文件描述符到高效命令行工作流

1. 项目概述:为什么重定向是命令行的效率倍增器?如果你在Linux命令行里混过一段时间,肯定遇到过这样的场景:想看看一个命令的输出,结果屏幕刷地一下滚过去几百行,关键信息一闪而过;或者想把一个…...

从IMU到UWB:拆解美国队长盾牌自主归位的嵌入式控制核心

1. 项目概述:从电影梦到工程挑战每个看过《美国队长》的人,大概都幻想过能像史蒂夫罗杰斯那样,潇洒地掷出那面标志性的振金盾牌,看着它在空中划出完美的弧线,击倒敌人后又精准地飞回手中。这不仅是超级英雄的浪漫&…...

TimerBlox:基于电流基准的硬件定时新方案,替代555与MCU

1. 项目概述:重新认识定时电路的设计范式在嵌入式系统、电源管理、电机控制乃至各类信号发生器的设计中,定时功能几乎无处不在。无论是生成一个精确的PWM波去调节LED亮度,还是产生一个可调的时钟信号驱动VCO,亦或是需要一个精准的…...

怎么限制用户上传到MongoDB GridFS的文件总容量

GridFS不支持全局容量配额,需在应用层实现配额校验:上传前聚合查询fs.files中指定用户的length总和,判断是否超限,且须防范并发写入导致的超限问题。GridFS 本身不提供全局容量配额机制MongoDB 的 GridFS 是一个文件分片存储规范&…...

智能车竞赛实战:从PID控制到图像识别的嵌入式系统开发全解析

1. 项目概述:一场硬核的嵌入式综合实战“飞思卡尔杯”智能车竞赛,这个名字对于很多电子、自动化、计算机相关专业的同学来说,绝对是一个如雷贯耳的存在。它不仅仅是一个比赛,更像是一个集机械、电子、控制、算法于一体的微型“工业…...

Simulink仿真PMSM时,那个神秘的‘4’和‘30/π’到底怎么来的?手把手带你算清楚

Simulink仿真PMSM时关键参数转换原理与实战解析 在永磁同步电机(PMSM)的Simulink仿真过程中,工程师们常常会遇到几个看似"神秘"的增益系数——特别是30/π和4这两个数值。这些参数并非随意设置,而是深植于电机物理本质与单位系统转换的数学表达…...

深度解析DsHidMini:开源项目实现Windows平台DualShock 3控制器用户态驱动

深度解析DsHidMini:开源项目实现Windows平台DualShock 3控制器用户态驱动 【免费下载链接】DsHidMini Virtual HID Mini-user-mode-driver for Sony DualShock 3 Controllers 项目地址: https://gitcode.com/gh_mirrors/ds/DsHidMini DsHidMini是一款基于Win…...

免费解锁B站大会员4K视频:Python下载器完整使用指南

免费解锁B站大会员4K视频:Python下载器完整使用指南 【免费下载链接】bilibili-downloader B站视频下载,支持下载大会员清晰度4K,持续更新中 项目地址: https://gitcode.com/gh_mirrors/bil/bilibili-downloader 还在为B站上的精彩视频…...

DLSS Swapper终极指南:5分钟快速上手游戏性能优化神器

DLSS Swapper终极指南:5分钟快速上手游戏性能优化神器 【免费下载链接】dlss-swapper 项目地址: https://gitcode.com/GitHub_Trending/dl/dlss-swapper 你是否曾为游戏中的DLSS版本过旧而烦恼?是否厌倦了手动下载、替换DLSS文件的繁琐过程&…...

《身体健康最重要》的内容入口:朴素标题如何连接听众

从内容传播角度看,《身体健康最重要》的入口非常朴素,也因此有记忆点。它不像复杂概念,而像日常里常被忽略的一句提醒。这类标题如果写成说教,很容易失去听众。更好的方式,是把它放回真实生活:熬夜后、忙到…...

如何提升SQL存储过程逻辑复用_封装通用存储过程函数

SQL Server无函数式存储过程,需用标量函数(单值计算)或表值函数(结果集)替代;标量函数禁用DML和非确定性函数,ITVF性能优于MSTVF;MySQL函数须声明DETERMINISTIC等属性;跨…...

基于MCP协议构建专属AI开发助手:从原理到实践

1. 项目概述:一个为开发者定制的MCP服务器最近在折腾AI应用开发,特别是想给Claude、Cursor这类智能助手增加一些“超能力”,让它们能直接操作我本地的开发环境。比如,让AI帮我直接运行单元测试、查看最近的Git提交、或者分析某个目…...

Parabolic视频下载工具:三步完成200+网站视频下载的终极方案

Parabolic视频下载工具:三步完成200网站视频下载的终极方案 【免费下载链接】Parabolic Download web video and audio 项目地址: https://gitcode.com/GitHub_Trending/pa/Parabolic 你是否还在为寻找一款简单易用、功能强大的视频下载工具而烦恼&#xff1…...

从NCDC到本地分析:一站式获取与处理全球气象站点数据

1. 全球气象数据获取的完整指南 第一次接触气象数据的朋友可能会被各种专业术语和数据格式搞得晕头转向。我刚开始做气象分析时,光是找数据就花了两周时间,下载下来的文件还经常打不开。今天我就把从数据获取到最终分析的完整流程梳理出来,帮…...

AI赋能效率革命:用ChatGPT+Markdown一键生成Xmind/ProcessOn专业流程图

1. 为什么需要AI辅助图表制作? 在日常工作和学习中,我们经常需要制作各种图表来梳理思路或展示信息。传统方式要么依赖专业软件操作(比如反复拖拽图形元件),要么需要手动调整格式排版,整个过程往往要花费半…...

GIS国土工具实战:从地类分析到坐标转换,一站式解决项目难题

1. GIS国土工具如何解决项目痛点 第一次接触国土整治项目时,我被各种数据格式搞得焦头烂额。早上9点收到甲方发来的50个地块的shp文件,下午3点就要提交带坐标的txt报备文件,中间还要做地类分析和影像核对。手动操作?光是想到要一个…...

告别“找不同”游戏:STANet的时空注意力如何让AI看懂遥感图中的建筑变化?

告别“找不同”游戏:STANet如何让AI像人类一样理解遥感图中的建筑变化? 想象一下,你面前有两张相隔数月的卫星照片,需要找出其中新建的楼盘或消失的农田。传统方法就像玩“找不同”游戏——逐像素对比颜色差异,稍有不慎…...

STM32L4实战:用RTC唤醒定时器实现33秒超长待机,实测功耗从52mA降到2.2mA

STM32L4超低功耗实战:从52mA到2.2mA的RTC唤醒优化全解析 当一块STM32L4开发板的功耗从52mA骤降到2.2mA,这不仅仅是数字的变化——它意味着智能穿戴设备的续航从1天延长到3周,工业传感器节点可以摆脱电源线的束缚,便携医疗设备的安…...

Odrive多线程架构实战:如何为你的无刷电机项目定制启动流程与状态机

Odrive多线程架构实战:如何为你的无刷电机项目定制启动流程与状态机 在无刷电机控制领域,Odrive凭借其开源特性和强大的性能表现,已成为众多硬件开发者的首选平台。但当你需要将这套系统集成到自己的机器人关节、CNC主轴或无人机推进系统中时…...

AI技能包管理:构建可复用的智能体技能生态

1. 项目概述:当AI技能也需要一个“缓存管家”最近在折腾AI应用开发,特别是基于LangChain、AutoGPT这类框架构建智能体时,有一个问题反复出现,让我头疼不已:技能(Skills)的管理与复用。简单来说&…...

跟着 MDN 学 HTML day_51:(深入理解 XPathEvaluator 接口)

在前端开发中,我们经常需要对 DOM 树进行复杂的节点查询。虽然 querySelector 和 querySelectorAll 已经能够满足大部分 CSS 选择器需求,但在某些场景下,我们需要更强大的查询能力,比如根据节点的文本内容查找、根据属性是否存在进…...

基于MCP协议与Graph API实现AI助手无缝集成Outlook邮箱

1. 项目概述与核心价值 最近在折腾AI工作流,发现一个挺有意思的项目: ajaya/outlook-app-mcp 。简单来说,这是一个能让你的AI助手(比如Claude Desktop、Cursor等支持MCP协议的客户端)直接读取和操作你Outlook邮箱的…...

模块四-数据转换与操作——24. 数据分箱

24. 数据分箱 1. 概述 数据分箱(Binning)是将连续变量离散化的过程,将数值范围划分为多个区间,每个区间称为一个"箱"。分箱常用于将连续变量转换为分类变量,便于分析和建模。 import pandas as pd import nu…...

别再乱接电阻了!STM32F407 SWD调试电路设计,从手册到实战的完整避坑指南

STM32F407 SWD调试电路设计:从芯片手册到工程实践的黄金法则 在嵌入式开发领域,调试接口的设计往往被当作"简单连线"而草率处理,直到某天你发现烧录器频繁断开连接、芯片无法识别,或是批量生产中出现随机性下载失败——…...

【工业视觉】基于序列图像动态特征提取的熔炼结晶过程建模与量化分析

1. 工业视觉在熔炼结晶过程中的核心价值 我第一次接触工业视觉在熔炼结晶中的应用是在2018年一个铜冶炼项目。当时工厂老师傅们还在用肉眼观察结晶状态,不仅效率低下,而且不同操作员的判断标准差异能达到15%。这种主观性带来的工艺波动直接影响了产品质量…...