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

QChart实战:从零构建动态数据波形图(含完整代码与注释)

1. 环境准备与基础配置在开始构建动态波形图之前我们需要先搭建好开发环境。这里假设你已经安装了Qt Creator我推荐使用5.15或更高版本因为这个版本对QChart的支持最完善。如果你还没安装可以直接去Qt官网下载开源版本。首先创建一个新的Qt Widgets Application项目。在.pro文件中第一件要做的事情就是添加charts模块支持。很多新手会忘记这一步导致编译时报找不到QChart相关的头文件。正确的做法是在.pro文件里加上这行QT charts widgets注意这里同时添加了widgets模块因为我们需要用QChartView来显示图表。我建议在项目创建时就加上这些配置避免后期因为模块缺失导致的编译错误。接下来在需要使用图表的头文件中添加必要的包含语句。我习惯把这些include集中放在一起#include QtCharts // 必须包含这个主头文件 #include QChartView #include QSplineSeries #include QDateTimeAxis #include QValueAxis这里有个小技巧如果你不确定该包含哪些头文件可以先写QtCharts这个总头文件编译器报错时再根据提示添加具体的子头文件。我在早期开发时经常这么做能快速定位缺少的头文件。2. 创建基础图表框架2.1 初始化图表对象让我们从创建一个基本的图表框架开始。首先在类的头文件中声明必要的成员变量private: QChart *m_chart; // 图表主对象 QSplineSeries *m_series; // 曲线系列 QDateTimeAxis *m_axisX; // X轴时间轴 QValueAxis *m_axisY; // Y轴数值轴在.cpp文件中我们来实现图表的初始化。我建议把这部分逻辑封装成一个独立的方法比如叫initChart()void MainWindow::initChart() { // 创建图表和系列 m_chart new QChart(); m_series new QSplineSeries(); m_axisX new QDateTimeAxis(); m_axisY new QValueAxis(); // 设置图表标题 m_chart-setTitle(实时数据波形图); m_chart-setTitleFont(QFont(Arial, 12, QFont::Bold)); // 将系列添加到图表 m_chart-addSeries(m_series); // 设置坐标轴 m_chart-addAxis(m_axisX, Qt::AlignBottom); m_chart-addAxis(m_axisY, Qt::AlignLeft); m_series-attachAxis(m_axisX); m_series-attachAxis(m_axisY); // 将图表设置到视图 QChartView *chartView new QChartView(m_chart); chartView-setRenderHint(QPainter::Antialiasing); setCentralWidget(chartView); // 如果是主窗口 }这段代码创建了一个最基本的图表框架但还没有设置具体的坐标范围和样式。在实际项目中我通常会把这些初始化步骤分开先搭好框架再逐步完善细节。2.2 配置坐标轴属性坐标轴的配置对波形图的显示效果至关重要。让我们来完善这部分// 配置X轴时间轴 QDateTime currentTime QDateTime::currentDateTime(); m_axisX-setTitleText(时间); m_axisX-setFormat(hh:mm:ss); m_axisX-setRange(currentTime, currentTime.addSecs(60)); // 默认显示60秒范围 m_axisX-setTickCount(6); // 刻度数量 // 配置Y轴数值轴 m_axisY-setTitleText(数值); m_axisY-setRange(0, 100); // 假设数值范围0-100 m_axisY-setTickCount(5); m_axisY-setLabelFormat(%.1f); // 显示1位小数 // 设置曲线样式 m_series-setName(数据曲线); m_series-setPen(QPen(Qt::blue, 2));这里有几个实用技巧时间轴的setFormat方法可以灵活控制时间显示格式比如只显示分钟和秒(mm:ss)setTickCount控制刻度数量但实际显示可能会略有调整Y轴的setLabelFormat支持printf风格的格式化字符串3. 实现动态数据更新3.1 定时器驱动数据更新动态波形图的核心就是实时更新数据。在Qt中最简单的实现方式是使用QTimer。首先在头文件中添加private slots: void updateData(); private: QTimer *m_timer;然后在初始化代码中添加定时器设置// 初始化定时器 m_timer new QTimer(this); connect(m_timer, QTimer::timeout, this, MainWindow::updateData); m_timer-start(100); // 每100毫秒更新一次updateData函数的实现是关键void MainWindow::updateData() { static qint64 startTime QDateTime::currentDateTime().toMSecsSinceEpoch(); qint64 currentTime QDateTime::currentDateTime().toMSecsSinceEpoch(); // 模拟数据 - 实际项目中替换为真实数据源 double value 50 30 * qSin((currentTime - startTime) / 1000.0); // 添加新数据点 m_series-append(currentTime, value); // 自动滚动保持显示最新的10秒数据 if (currentTime - startTime 10000) { m_axisX-setRange(QDateTime::fromMSecsSinceEpoch(currentTime - 10000), QDateTime::fromMSecsSinceEpoch(currentTime)); } // 限制数据点数量防止内存无限增长 if (m_series-count() 200) { m_series-remove(0); } }这段代码实现了几个关键功能模拟生成正弦波数据实际项目替换为真实数据自动滚动时间轴始终保持显示最新10秒数据限制数据点数量避免内存泄漏3.2 性能优化技巧当数据更新频率很高时可能会遇到性能问题。这里分享几个我总结的优化技巧减少重绘频率可以设置m_chart-setAnimationOptions(QChart::NoAnimation)来禁用动画效果批量添加数据点如果一次需要添加多个点可以先调用m_series-clear()再一次性添加所有点使用更轻量的系列类型对于高频数据QSplineSeries可能会成为性能瓶颈可以尝试改用QLineSeries控制更新频率适当降低定时器间隔比如从100ms调整为200ms// 在初始化时添加这些优化设置 m_chart-setAnimationOptions(QChart::NoAnimation); m_chart-setMargins(QMargins(0, 0, 0, 0)); // 减少边距4. 高级定制与样式美化4.1 主题与颜色定制QChart提供了多种内置主题也可以完全自定义样式。下面是一些常用设置// 使用内置暗色主题 m_chart-setTheme(QChart::ChartThemeDark); // 或者完全自定义颜色 m_chart-setBackgroundBrush(QBrush(Qt::transparent)); // 透明背景 m_chart-setTitleBrush(QBrush(Qt::white)); // 标题颜色 m_axisX-setLabelsColor(Qt::white); // X轴标签颜色 m_axisY-setLabelsColor(Qt::white); // Y轴标签颜色 // 网格线设置 m_axisX-setGridLineVisible(true); m_axisY-setGridLineVisible(true); m_axisX-setGridLinePen(QPen(QColor(80, 80, 80), 1, Qt::DotLine)); m_axisY-setGridLinePen(QPen(QColor(80, 80, 80), 1, Qt::DotLine)); // 图例设置 m_chart-legend()-setVisible(true); m_chart-legend()-setAlignment(Qt::AlignBottom); m_chart-legend()-setLabelColor(Qt::white);4.2 多曲线支持实际项目中经常需要同时显示多条曲线。下面是添加第二条曲线的方法// 在头文件中添加 private: QSplineSeries *m_series2; // 在初始化代码中添加 m_series2 new QSplineSeries(); m_series2-setName(参考曲线); m_series2-setPen(QPen(Qt::red, 2, Qt::DashLine)); m_chart-addSeries(m_series2); m_series2-attachAxis(m_axisX); m_series2-attachAxis(m_axisY); // 在updateData中更新第二条曲线 double refValue 50 10 * qCos((currentTime - startTime) / 1000.0); m_series2-append(currentTime, refValue); if (m_series2-count() 200) { m_series2-remove(0); }4.3 交互功能增强为了让波形图更具交互性可以添加这些功能// 允许鼠标缩放 chartView-setRubberBand(QChartView::RectangleRubberBand); // 右键复位视图 chartView-setInteractive(true); chartView-setDragMode(QGraphicsView::ScrollHandDrag);还可以添加十字线效果显示当前鼠标位置的值// 在头文件中添加 private: QGraphicsSimpleTextItem *m_coordX; QGraphicsSimpleTextItem *m_coordY; // 在初始化代码中添加 m_coordX new QGraphicsSimpleTextItem(m_chart); m_coordY new QGraphicsSimpleTextItem(m_chart); m_coordX-setPos(100, 100); m_coordY-setPos(100, 120); m_coordX-setBrush(QBrush(Qt::white)); m_coordY-setBrush(QBrush(Qt::white)); // 连接鼠标移动信号 connect(chartView, QChartView::mouseMove, [](QMouseEvent *event) { QPointF point chartView-chart()-mapToValue(event-pos()); m_coordX-setText(QString(X: %1).arg(QDateTime::fromMSecsSinceEpoch(point.x()).toString(hh:mm:ss))); m_coordY-setText(QString(Y: %1).arg(point.y(), 0, f, 2)); });5. 实际项目中的经验分享在真实项目中应用QChart绘制波形图时有几个常见问题需要注意时间同步问题在多线程环境下确保数据采集线程和UI线程的时间戳同步。我通常会在数据采集线程获取数据时立即记录时间戳而不是在UI线程获取当前时间。性能瓶颈当需要显示大量数据点时比如上万点QChart的性能会明显下降。这种情况下我有两个解决方案实现数据降采样只显示关键点考虑使用更专业的绘图库如QCustomPlot内存管理长时间运行的应用程序要注意及时清理不再需要的数据点避免内存泄漏。我建议设置一个合理的点数上限像前面的例子中限制为200点。跨平台兼容性不同平台下QChart的渲染效果可能略有差异特别是在嵌入式Linux平台上。测试时一定要在实际目标平台上验证显示效果。异常处理数据源可能会产生异常值如NaN或无穷大这些值会导致绘图异常。我通常在添加数据点前先进行检查if (!qIsNaN(value) qIsFinite(value)) { m_series-append(timeStamp, value); }最后分享一个我在项目中遇到的真实案例有次波形图突然停止更新调试发现是因为数据源的时间戳出现了回跳可能是NTP同步导致。后来我在代码中添加了时间戳校验逻辑确保新数据的时间戳总是大于最后一点的时间戳问题才得以解决。

相关文章:

QChart实战:从零构建动态数据波形图(含完整代码与注释)

1. 环境准备与基础配置 在开始构建动态波形图之前,我们需要先搭建好开发环境。这里假设你已经安装了Qt Creator,我推荐使用5.15或更高版本,因为这个版本对QChart的支持最完善。如果你还没安装,可以直接去Qt官网下载开源版本。 首…...

解决NextCloud无法挂载SMB/CIFS共享:smbclient缺失的完整安装指南

1. 为什么NextCloud需要smbclient支持 如果你正在使用NextCloud搭建私有云存储,可能会遇到一个常见问题:无法挂载SMB/CIFS共享存储。这个问题通常会在管理后台的"外部存储"设置页面出现错误提示,核心原因就是缺少smbclient组件。 S…...

告别文件传输烦恼:用aliyunpan快传链接实现秒级大文件分享

告别文件传输烦恼:用aliyunpan快传链接实现秒级大文件分享 【免费下载链接】aliyunpan 阿里云盘命令行客户端,支持JavaScript插件,支持同步备份功能。 项目地址: https://gitcode.com/GitHub_Trending/ali/aliyunpan 你是否也曾经历过…...

【实战指南】系统变量编辑权限问题全解析

1. 系统变量编辑权限问题解析 最近在帮同事调试开发环境时,遇到一个典型问题:明明已经用管理员账号登录,却死活改不了系统环境变量。这让我想起自己刚接触Windows系统时踩过的坑,今天就把这些经验系统梳理一下。 系统变量本质上是…...

SurfaceView视觉优化实战:圆角与渐变蒙层的完美结合

1. SurfaceView视觉优化的核心价值 在Android开发中,SurfaceView因其独特的双缓冲机制和独立的绘图线程,成为视频播放、游戏渲染等高性能场景的首选组件。但原生SurfaceView的直角边框和单调的呈现方式,常常与现代化UI设计语言格格不入。我在…...

foobox-cn:让foobar2000从工具变身艺术品的终极美化方案

foobox-cn:让foobar2000从工具变身艺术品的终极美化方案 【免费下载链接】foobox-cn DUI 配置 for foobar2000 项目地址: https://gitcode.com/GitHub_Trending/fo/foobox-cn 你是否还在忍受foobar2000那过于朴素的默认界面?是否觉得功能强大的播…...

XCOM 2模组管理的终极解决方案:Alternative Mod Launcher完整指南

XCOM 2模组管理的终极解决方案:Alternative Mod Launcher完整指南 【免费下载链接】xcom2-launcher The Alternative Mod Launcher (AML) is a replacement for the default game launchers from XCOM 2 and XCOM Chimera Squad. 项目地址: https://gitcode.com/g…...

从‘基’到‘坐标变换’:用Python和NumPy手把手理解线性空间的‘换地图’操作

从‘基’到‘坐标变换’:用Python和NumPy手把手理解线性空间的‘换地图’操作 想象一下,你正在使用导航软件规划路线。同一个地点,在高德地图和百度地图上显示的坐标可能完全不同——这就像线性代数中的基变换。本文将用Python代码和可视化手…...

嵌入式WiFi开发 | 基于wireless_tools的交叉编译实战与移植指南

1. 嵌入式WiFi开发入门:为什么需要wireless_tools? 在嵌入式Linux开发中,网络连接能力往往是刚需。想象一下你的智能家居设备需要自动连接路由器,或者工业传感器需要通过WiFi上传数据——这些都离不开可靠的无线网络配置工具。这就…...

太阳能电池阵列监测实战:用AMC1301搞定200V共模电压下的单体电压采集

太阳能电池阵列单体电压监测:基于AMC1301的高压隔离采集方案设计指南 光伏电站的电池阵列通常由数十至数百块单体电池串联组成,系统电压可达600-1500V。在这种高压堆叠场景下,如何准确监测每块单体电池的电压(通常仅0.5-0.7V&…...

MoveIt新手避坑:Gazebo仿真时遇到‘Unable to identify controllers‘报错,检查这个launch文件就对了

MoveIt新手避坑:Gazebo仿真时遇到Unable to identify controllers报错解决方案 当你第一次尝试在Gazebo中运行MoveIt控制机械臂时,看到终端弹出鲜红的报错信息"Unable to identify any set of controllers that can actuate the specified joints&q…...

探索ArtPlayer:如何通过轻量高效的HTML5视频引擎实现全场景适配播放体验

探索ArtPlayer:如何通过轻量高效的HTML5视频引擎实现全场景适配播放体验 【免费下载链接】ArtPlayer :art: ArtPlayer.js is a modern and full featured HTML5 video player 项目地址: https://gitcode.com/gh_mirrors/ar/ArtPlayer 在数字内容爆发的时代&a…...

OptiScaler终极指南:3步解锁跨平台超分辨率技术,让所有显卡享受DLSS级画质提升

OptiScaler终极指南:3步解锁跨平台超分辨率技术,让所有显卡享受DLSS级画质提升 【免费下载链接】OptiScaler DLSS replacement for AMD/Intel/Nvidia cards with multiple upscalers (XeSS/FSR2/DLSS) 项目地址: https://gitcode.com/GitHub_Trending/…...

ComfyUI插件避坑指南:国内用户如何解决模型下载和安装问题

ComfyUI插件避坑指南:国内用户如何解决模型下载和安装问题 如果你是一名国内用户,想要使用ComfyUI的插件来提升工作效率,那么你可能会遇到一些令人头疼的问题。模型下载缓慢、安装报错、依赖冲突...这些问题不仅浪费时间,还容易让…...

手把手教你用ZPL指令在Zebra打印机上打印动态条码(附完整代码示例)

手把手教你用ZPL指令在Zebra打印机上打印动态条码(附完整代码示例) 在物流仓储、零售结算和智能制造场景中,自动生成并打印条码标签是提升作业效率的关键环节。Zebra打印机凭借其工业级稳定性和ZPL语言的高效指令集,成为行业标配…...

Cesium交互绘图避坑指南:从CallbackProperty到CustomDataSource的完整流程

Cesium交互绘图避坑指南:从CallbackProperty到CustomDataSource的完整流程 在三维地理信息可视化领域,Cesium凭借其强大的渲染能力和丰富的API接口,已成为开发者构建交互式地图应用的首选工具。然而,当涉及动态绘图功能时&#xf…...

OpenClaw日志分析:Qwen3-32B每日自动汇总服务器异常事件

OpenClaw日志分析:Qwen3-32B每日自动汇总服务器异常事件 1. 为什么需要自动化日志分析 作为一名运维工程师,我每天早晨的第一项工作就是检查服务器日志。Nginx的错误日志、系统内核日志、应用服务的异常输出……这些文件分散在不同的目录,格…...

Ubuntu22.04桌面版root登录避坑指南:从密码设置到SSH远程连接完整流程

Ubuntu 22.04桌面版root权限全流程实战:从密码安全到SSH调优 刚接触Ubuntu桌面环境时,很多开发者会遇到这样的困境:图形界面操作需要频繁输入sudo密码,而某些系统级配置又必须使用root账户。本文将带你用工程师思维解决这个痛点&a…...

Hive【从SQL到MapReduce:核心架构与执行引擎深度解析】

1. Hive的核心角色:SQL到分布式计算的翻译官 第一次接触Hive时,很多人会疑惑:为什么要在Hadoop生态中引入这样一个"类SQL"工具?这要从大数据处理的痛点说起。想象你面前有一本百万页的百科全书,现在需要统计…...

自然语言生成:为AI原生应用注入新活力

自然语言生成:为AI原生应用注入新活力 关键词:自然语言生成(NLG)、AI原生应用、大语言模型、文本生成、多模态交互 摘要:自然语言生成(NLG)是AI领域的“语言魔法”,能让机器像人类一…...

三、从零解析Franka ROS2控制器:以关节位置控制为例

1. Franka机械臂与ROS2控制器基础 如果你刚接触机器人控制,Franka机械臂搭配ROS2绝对是个不错的起点。Franka Emika机械臂以其高精度和易用性著称,而ROS2作为机器人操作系统的最新版本,提供了更强大的实时性和分布式能力。我第一次用Franka做…...

千问3.5-27B效果展示:手写笔记图片→文字转录→知识点归类→复习卡片生成

千问3.5-27B效果展示:手写笔记图片→文字转录→知识点归类→复习卡片生成 1. 模型核心能力概览 Qwen3.5-27B作为一款视觉多模态理解模型,在知识处理领域展现出独特优势。它不仅能理解图片内容,还能对信息进行深度加工。本次重点展示其从手写…...

别再死磕分布式事务了!用MySQL+RabbitMQ手撸一个本地消息表,搞定订单库存一致性问题

轻量级数据一致性实战:基于MySQL与RabbitMQ的本地消息表设计 在电商系统开发中,订单创建与库存扣减的原子性操作一直是技术难点。传统单体架构下的数据库事务无法跨越服务边界,而引入分布式事务框架又往往带来额外的复杂性和性能损耗。本文将…...

如何用Java处理地震波?信号滤波算法

常用的地震波信号滤波算法包括傅里叶转换(fft)与频域滤波器、fir滤波器、iir滤波器和中值滤波器一起。. 通过将时域信号转换为频域,java可以通过apache实现特定频率组件的操作 commons math库中的fastfouriertransformer类实现;2.…...

FPGA新手别怕!Vivado 2023.1里用DDS IP核生成1MHz正弦波,保姆级图文配置+仿真

FPGA实战:从零开始用Vivado配置DDS IP核生成精准波形 第一次打开Vivado的IP Catalog界面时,满屏的参数选项确实容易让人望而生畏。但别担心,DDS(直接数字频率合成)IP核其实比你想象的要友好得多。作为FPGA数字信号处理…...

告别笨重线性电源!用TL494打造高效BUCK模块,给你的老旧设备供电或做充电器

用TL494打造高效BUCK模块:老设备供电与智能充电的终极解决方案 老旧实验室设备嗡嗡作响的线性电源,不仅效率低下,发热严重,还占据宝贵的工作台空间。而一块基于TL494的高效BUCK模块,可以彻底改变这一局面。本文将带你…...

AppleRa1n开源工具:iOS 15-16激活锁绕过完整解决方案

AppleRa1n开源工具:iOS 15-16激活锁绕过完整解决方案 【免费下载链接】applera1n icloud bypass for ios 15-16 项目地址: https://gitcode.com/gh_mirrors/ap/applera1n 激活锁(Activation Lock)作为iOS设备的重要安全机制&#xff0…...

ASan实战:5种常见内存错误诊断与修复指南(附GCC/Clang编译命令)

ASan实战:5种常见内存错误诊断与修复指南(附GCC/Clang编译命令) 在C/C开发中,内存错误如同潜伏的暗礁,随时可能让程序沉没。AddressSanitizer(ASan)作为Google推出的内存错误检测工具&#xff…...

51单片机(九)—— 数码管动态扫描原理与实现

1. 数码管动态扫描原理揭秘 第一次接触多位数码管显示时,我盯着电路板百思不得其解:明明只有8个数据引脚,怎么能同时控制8位数码管显示不同内容?直到理解了动态扫描原理,才恍然大悟这背后的精妙设计。动态扫描本质上是…...

Win11环境实测:用C# EtherCAT库控制伺服电机,从TwinCAT配置到pcap抓包全流程避坑

Win11环境下的EtherCAT实战:C#控制伺服电机全流程解析 在工业自动化领域,EtherCAT凭借其高速、实时的特性已成为运动控制系统的首选协议之一。本文将带你深入Windows 11环境下使用C#开发EtherCAT主站的全过程,从TwinCAT配置到实际控制伺服电机…...