QT数据可视化框架编程实战之三维曲面图 实时变化的三维曲面图 补天云QT技术培训专家

本文主要是介绍QT数据可视化框架编程实战之三维曲面图 实时变化的三维曲面图 补天云QT技术培训专家,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

QT数据可视化框架编程实战之三维曲面图 实时变化的三维曲面图 补天云QT技术培训专家

简介

本文将介绍QT数据可视化框架编程实战之三维曲面图,本文通过构造一个数据实时变化的三维曲面图的应用实例来展示QT数据可视化框架的数据展示能力,同时还将给出这个应用实例的C++源码和QML源码。

文章目录

  • QT三维曲面图展示实时变化数据的运行效果
  • QT三维曲面图展示实时变化数据的设计思路
  • QT三维曲面图展示实时变化数据的C++主程序
  • QT三维曲面图展示实时变化数据的数据源C++源码
  • QT三维曲面图展示实时变化数据的QML源码

正文

QT三维曲面图展示实时变化数据的运行效果

QT数据可视化框架中的三维曲面图,展示实时变化的序列数据的运行效果截图如下所示:

在这里插入图片描述

运行效果的GIF动画如下所示:

在这里插入图片描述

运行效果和对应的源码浏览的视频如下所示:

QT数据可视化框架编程实战之三维曲面图 实时变化的三维曲面图 补天云QT技术培训专家

视频链接如下所示:

QT数据可视化框架编程实战之三维曲面图 实时变化的三维曲面图 补天云QT技术培训专家

QT三维曲面图展示实时变化数据的设计思路

QT数据可视化框架展示实时变化的三维曲面图,首先想到的就是动态的实时生成曲面图序列的数据点。考虑到数据点的数量可能比较多,那么为了提升运行效率,可以将实时生成曲面图序列的数据点的功能使用C++来实现,然后使用QT三维曲面图的QML组件来展示这些序列的数据,从而实现实时变化数据的展示。

为了不断生成变化的数据点,可以考虑使用QML定时器来定时更新数据点;为了及时的展示出变化之后的数据点,可以考虑使用定时器来定时刷新QT三维曲面图。

QT三维曲面图展示实时变化数据的C++主程序

butianyun.cpp文件如下所示:


int main(int argc, char *argv[])
{qputenv("QSG_RHI_BACKEND", "opengl");QGuiApplication app(argc, argv);qmlRegisterType<ButianyunDataSource>("ButianyunDataSource", 1, 0, "ButianyunDataSource");QQmlApplicationEngine engine;const QUrl url(u"qrc:/Butianyun3DSurface/Butianyun3DSurface.qml"_qs);QObject::connect(&engine, &QQmlApplicationEngine::objectCreationFailed,&app, []() { QCoreApplication::exit(-1); },Qt::QueuedConnection);engine.load(url);return app.exec();
}

QT三维曲面图展示实时变化数据的数据源C++源码

数据源使用C++代码来生成。
ButianyunDataSource.cpp文件和ButianyunDataSource.h文件实现了数据源生成功能。

(备注:本文广泛参考了QT框架提供的QT可视化框架的例程的源码)

生成新的数据点的C++代码如下所示:

void ButianyunDataSource::generateData(int cacheCount, int rowCount, int columnCount,float xMin, float xMax,float yMin, float yMax,float zMin, float zMax)
{if (!cacheCount || !rowCount || !columnCount){return;}clearData();// 重建缓存数据m_data.resize(cacheCount);for (int i = 0; i < cacheCount; ++i){QSurfaceDataArray &array = m_data[i];array.reserve(rowCount);for (int j = 0; j < rowCount; ++j){array.append(new QSurfaceDataRow(columnCount));}}float xRange = xMax - xMin;float yRange = yMax - yMin;float zRange = zMax - zMin;int cacheIndexStep = columnCount / cacheCount;float cacheStep = float(cacheIndexStep) * xRange / float(columnCount);// 每次生成数据时一次性生成多个序列数据// 遍历缓存:缓存中的每一个序列数据就是用于某一次展示,也就是一个序列auto *generator = QRandomGenerator::global();for (int i = 0; i < cacheCount; ++i){QSurfaceDataArray &cache = m_data[i];float cacheXAdjustment = cacheStep * i;float cacheIndexAdjustment = cacheIndexStep * i;for (int j = 0; j < rowCount; ++j){QSurfaceDataRow &row = *(cache[j]);float rowMod = (float(j)) / float(rowCount);float yRangeMod = yRange * rowMod;float zRangeMod = zRange * rowMod;float z = zRangeMod + zMin;qreal rowColWaveAngleMul = M_PI * M_PI * rowMod;float rowColWaveMul = yRangeMod * 0.2f;for (int k = 0; k < columnCount; k++){float colMod = (float(k)) / float(columnCount);float xRangeMod = xRange * colMod;float x = xRangeMod + xMin + cacheXAdjustment;float colWave = float(qSin((2.0 * M_PI * colMod) - (1.0 / 2.0 * M_PI)) + 1.0);float y = (colWave * ((float(qSin(rowColWaveAngleMul * colMod) + 1.0))))* rowColWaveMul+ generator->bounded(0.03f) * yRangeMod+ generator->bounded(0.03f) * (zRange - zRangeMod);int index = k + cacheIndexAdjustment;if (index >= columnCount){index -= columnCount;x -= xRange;}row[index] = QVector3D(x, y, z);}}}
}

刷新QT三维曲面图的序列的C++代码如下所示:


void ButianyunDataSource::update(QSurface3DSeries *series)
{if (series && !m_data.isEmpty()){// 缓存数据的索引序号if (++m_index >= m_data.size()){m_index = 0;}const QSurfaceDataArray &array = m_data.at(m_index);int newRowCount = array.size();int newColumnCount = array.at(0)->size();// 重新序列重构数据if (!m_resetArray || series->dataProxy()->rowCount() != newRowCount|| series->dataProxy()->columnCount() != newColumnCount){m_resetArray = new QSurfaceDataArray();m_resetArray->reserve(newRowCount);for (int i = 0; i < newRowCount; ++i){m_resetArray->append(new QSurfaceDataRow(newColumnCount));}}// 从缓存中拷贝数据到序列重构数据中for (int i = 0; i < newRowCount; ++i){const QSurfaceDataRow &sourceRow = *(array.at(i));QSurfaceDataRow &row = *(*m_resetArray)[i];std::copy(sourceRow.cbegin(), sourceRow.cend(), row.begin());}// 通知代理对象:序列的数据已经变化了,以便刷新界面series->dataProxy()->resetArray(m_resetArray);}
}

QT三维曲面图展示实时变化数据的QML源码

Butianyun3DSurface.qml文件源码如下所示:

Window {width: 1920height: 1000visible: truetitle: qsTr("QT数据可视化  补天云QT系列视频课程 补天云QT技术培训专家")//数据源ButianyunDataSource {id: dataSource}//QML三维曲面Surface3D {id: surfaceGraphanchors.fill: parent//QML三维曲面的序列Surface3DSeries {id: surfaceSeriesdrawMode: Surface3DSeries.DrawSurfaceflatShadingEnabled: falseitemLabelFormat: "@xLabel, @zLabel: @yLabel"itemLabelVisible: false}shadowQuality: AbstractGraph3D.ShadowQualityNoneselectionMode: AbstractGraph3D.SelectionSlice | AbstractGraph3D.SelectionItemAndColumntheme: Theme3D {type: Theme3D.ThemeIsabellebackgroundEnabled: false}scene.activeCamera.cameraPreset: Camera3D.CameraPresetBehindaxisX.labelFormat: "%d ms"axisY.labelFormat: "%d mm"axisZ.labelFormat: "%d mm"axisX.min: 0axisY.min: 0axisZ.min: 0axisX.max: 1000axisY.max: 1000axisZ.max: 1000axisX.segmentCount: 10axisY.segmentCount: 10axisZ.segmentCount: 10measureFps: truerenderingMode: AbstractGraph3D.RenderDirectToBackground//首次产生数据Component.onCompleted: generateData();}property int cacheCount: 15property int fps:  30property int rowCount:  30property int columnCount: 30//产生数据的包装函数function generateData() {dataSource.generateData(cacheCount, rowCount,columnCount,surfaceGraph.axisX.min, surfaceGraph.axisX.max,surfaceGraph.axisY.min, surfaceGraph.axisY.max,surfaceGraph.axisZ.min, surfaceGraph.axisZ.max);}//定时器:定时重新产生数据Timer {interval: 1000running: truerepeat: trueonTriggered: {generateData();}}//定时器:定时刷新曲面界面Timer {interval: 1000 / fpsrunning: truerepeat: trueonTriggered: {dataSource.update(surfaceSeries);}}}

总结

本文介绍了QT数据可视化框架编程实战之三维曲面图,本文通过构造一个数据实时变化的三维曲面图的应用实例来展示QT数据可视化框架的数据展示能力,同时还给出了这个应用实例的C++源码和QML源码。

本文介绍的技术路线也存在进一步优化改进的空间,比如本文使用C++代码来实时生成数据点,使用的是CPU的运算能力,如果能使用GLSL之类的shader代码来实时生成数据点,则可以充分发挥GPU的运算能力,将可能更进一步提升三维曲面图应用程序的整体性能。

如果您认为这篇文章对您有所帮助,请您一定立即点赞+喜欢+收藏,本文作者将能从您的点赞+喜欢+收藏中获取到创作新的好文章的动力。如果您认为作者写的文章还有一些参考价值,您也可以关注这篇文章的作者。

这篇关于QT数据可视化框架编程实战之三维曲面图 实时变化的三维曲面图 补天云QT技术培训专家的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/341312

相关文章

详谈redis跟数据库的数据同步问题

《详谈redis跟数据库的数据同步问题》文章讨论了在Redis和数据库数据一致性问题上的解决方案,主要比较了先更新Redis缓存再更新数据库和先更新数据库再更新Redis缓存两种方案,文章指出,删除R... 目录一、Redis 数据库数据一致性的解决方案1.1、更新Redis缓存、删除Redis缓存的区别二

Redis事务与数据持久化方式

《Redis事务与数据持久化方式》该文档主要介绍了Redis事务和持久化机制,事务通过将多个命令打包执行,而持久化则通过快照(RDB)和追加式文件(AOF)两种方式将内存数据保存到磁盘,以防止数据丢失... 目录一、Redis 事务1.1 事务本质1.2 数据库事务与redis事务1.2.1 数据库事务1.

基于Qt开发一个简单的OFD阅读器

《基于Qt开发一个简单的OFD阅读器》这篇文章主要为大家详细介绍了如何使用Qt框架开发一个功能强大且性能优异的OFD阅读器,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 目录摘要引言一、OFD文件格式解析二、文档结构解析三、页面渲染四、用户交互五、性能优化六、示例代码七、未来发展方向八、结论摘要

Golang使用minio替代文件系统的实战教程

《Golang使用minio替代文件系统的实战教程》本文讨论项目开发中直接文件系统的限制或不足,接着介绍Minio对象存储的优势,同时给出Golang的实际示例代码,包括初始化客户端、读取minio对... 目录文件系统 vs Minio文件系统不足:对象存储:miniogolang连接Minio配置Min

Oracle Expdp按条件导出指定表数据的方法实例

《OracleExpdp按条件导出指定表数据的方法实例》:本文主要介绍Oracle的expdp数据泵方式导出特定机构和时间范围的数据,并通过parfile文件进行条件限制和配置,文中通过代码介绍... 目录1.场景描述 2.方案分析3.实验验证 3.1 parfile文件3.2 expdp命令导出4.总结

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

更改docker默认数据目录的方法步骤

《更改docker默认数据目录的方法步骤》本文主要介绍了更改docker默认数据目录的方法步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1.查看docker是否存在并停止该服务2.挂载镜像并安装rsync便于备份3.取消挂载备份和迁

不删数据还能合并磁盘? 让电脑C盘D盘合并并保留数据的技巧

《不删数据还能合并磁盘?让电脑C盘D盘合并并保留数据的技巧》在Windows操作系统中,合并C盘和D盘是一个相对复杂的任务,尤其是当你不希望删除其中的数据时,幸运的是,有几种方法可以实现这一目标且在... 在电脑生产时,制造商常为C盘分配较小的磁盘空间,以确保软件在运行过程中不会出现磁盘空间不足的问题。但在

你的华为手机升级了吗? 鸿蒙NEXT多连推5.0.123版本变化颇多

《你的华为手机升级了吗?鸿蒙NEXT多连推5.0.123版本变化颇多》现在的手机系统更新可不仅仅是修修补补那么简单了,华为手机的鸿蒙系统最近可是动作频频,给用户们带来了不少惊喜... 为了让用户的使用体验变得很好,华为手机不仅发布了一系列给力的新机,还在操作系统方面进行了疯狂的发力。尤其是近期,不仅鸿蒙O

如何评价Ubuntu 24.04 LTS? Ubuntu 24.04 LTS新功能亮点和重要变化

《如何评价Ubuntu24.04LTS?Ubuntu24.04LTS新功能亮点和重要变化》Ubuntu24.04LTS即将发布,带来一系列提升用户体验的显著功能,本文深入探讨了该版本的亮... Ubuntu 24.04 LTS,代号 Noble NumBAT,正式发布下载!如果你在使用 Ubuntu 23.