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

别再手动改Excel了!用QT的QFile和QTextStream搞定CSV读写(附线程安全锁)

用QT实现高效CSV自动化处理告别Excel手工操作在数据处理领域CSV文件因其简单通用而广受欢迎。作为C开发者我们经常需要处理各种数据导出、日志记录等任务。传统做法可能是手动操作Excel但这在自动化系统中显然行不通。QT框架提供的QFile和QTextStream类配合QMutex线程安全机制能够构建出稳定可靠的CSV处理模块。1. CSV处理的核心优势与应用场景CSV(Comma-Separated Values)文件本质上是以特定格式组织的纯文本具有几个显著优势跨平台兼容性几乎所有数据处理软件都能打开CSV包括Excel、Numbers、文本编辑器等轻量级相比Excel文件CSV体积更小处理速度更快易于程序处理简单的文本格式便于程序读写和解析在实际开发中CSV处理常见于以下场景系统日志记录持续写入程序运行状态、错误信息数据采集导出定期保存传感器读数、实验数据配置批量处理导入导出大量配置参数报表生成自动创建可供Excel打开的数据报表// 基本CSV文件结构示例 DateTime,EventType,Message 2023/08/15 14:30:22,INFO,System started 2023/08/15 14:35:18,WARNING,Disk space low2. QT CSV处理核心组件解析QT提供了一套完整的文件IO解决方案特别适合处理CSV这类文本文件。2.1 QFile文件操作基础类QFile是QT中用于文件读写的核心类提供了丰富的文件操作接口open()以指定模式打开文件close()关闭文件read()/write()基础读写操作exists()检查文件是否存在提示始终确保文件操作后调用close()或使用RAII风格的QFile对象管理2.2 QTextStream文本处理利器QTextStream大大简化了文本文件的读写操作QFile file(data.csv); if(file.open(QIODevice::WriteOnly | QIODevice::Text)) { QTextStream out(file); out Name,Age,Occupation\n; out John Doe,30,Engineer\n; file.close(); }关键特性包括自动处理文本编码转换支持类似C标准流的操作符(和)提供便捷的读写整行、全部内容等方法2.3 QMutex线程安全保证在多线程环境中操作文件时必须考虑线程安全问题。QMutex提供了简单的互斥锁机制static QMutex csvMutex; void writeToCSV(const QString data) { QMutexLocker locker(csvMutex); // 自动加锁函数返回时解锁 QFile file(data.csv); // ...文件操作... }3. 构建健壮的CSV处理模块让我们从零开始构建一个完整的CSV处理工具类包含以下功能自动创建CSV文件和目录线程安全写入高效读取解析错误处理机制3.1 CSV文件初始化良好的初始化是可靠性的基础。以下代码展示了如何安全地创建CSV文件和目录class CSVHandler { public: explicit CSVHandler(const QString baseDir C:/CSVData) { // 确保目录存在 QDir dir; if (!dir.exists(baseDir)) { dir.mkpath(baseDir); } // 生成带时间戳的文件名 QString timestamp QDateTime::currentDateTime().toString(yyyyMMdd_hhmmss); m_filePath QString(%1/data_%2.csv).arg(baseDir).arg(timestamp); // 初始化文件头 initCSVHeader(); } private: QString m_filePath; void initCSVHeader() { QMutexLocker locker(m_mutex); QFile file(m_filePath); if (file.open(QIODevice::WriteOnly | QIODevice::Text)) { QTextStream out(file); out Timestamp,EventType,Message\n; file.close(); } } static QMutex m_mutex; };3.2 线程安全写入实现数据写入需要考虑多种实际场景追加新数据错误处理性能优化bool CSVHandler::appendRecord(const QString eventType, const QString message) { QMutexLocker locker(m_mutex); QFile file(m_filePath); if (!file.open(QIODevice::WriteOnly | QIODevice::Append | QIODevice::Text)) { qWarning() Failed to open CSV file for writing: file.errorString(); return false; } QTextStream out(file); QString timestamp QDateTime::currentDateTime().toString(yyyy/MM/dd hh:mm:ss); out timestamp , eventType , message \n; file.close(); return true; }3.3 CSV数据读取与解析读取CSV时需要考虑处理空行解析带逗号的内容错误恢复QVectorCSVRecord CSVHandler::readAllRecords() const { QMutexLocker locker(m_mutex); QVectorCSVRecord records; QFile file(m_filePath); if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) { qWarning() Failed to open CSV file for reading: file.errorString(); return records; } QTextStream in(file); bool firstLine true; while (!in.atEnd()) { QString line in.readLine(); if (line.isEmpty()) continue; if (firstLine) { firstLine false; continue; // 跳过标题行 } QStringList parts line.split(,); if (parts.size() 3) { CSVRecord record; record.timestamp parts[0]; record.eventType parts[1]; record.message parts[2]; records.append(record); } } file.close(); return records; }4. 高级应用与性能优化掌握了基础操作后我们可以进一步优化CSV处理模块。4.1 批量写入优化频繁的小数据写入会影响性能实现批量写入可以显著提高效率void CSVHandler::appendRecords(const QVectorCSVRecord records) { if (records.isEmpty()) return; QMutexLocker locker(m_mutex); QFile file(m_filePath); if (!file.open(QIODevice::WriteOnly | QIODevice::Append | QIODevice::Text)) { return; } QTextStream out(file); for (const auto record : records) { out record.timestamp , record.eventType , record.message \n; } file.close(); }4.2 内存映射文件处理对于超大CSV文件可以使用内存映射提高读取速度QStringList CSVHandler::fastReadLines(int maxLines) const { QMutexLocker locker(m_mutex); QStringList lines; QFile file(m_filePath); if (!file.open(QIODevice::ReadOnly)) { return lines; } uchar *memory file.map(0, file.size()); if (memory) { QByteArray data QByteArray::fromRawData(reinterpret_castconst char*(memory), file.size()); QTextStream in(data); int count 0; while (!in.atEnd() (maxLines 0 || count maxLines)) { lines in.readLine(); count; } file.unmap(memory); } file.close(); return lines; }4.3 CSV与数据结构转换实际应用中我们经常需要在CSV和数据结构间转换// 从数据结构生成CSV行 QString CSVHandler::recordToCSVLine(const CSVRecord record) const { return QString(\%1\,\%2\,\%3\) .arg(record.timestamp) .arg(record.eventType) .arg(record.message); } // 从CSV行解析数据结构 CSVRecord CSVHandler::csvLineToRecord(const QString line) const { CSVRecord record; QStringList parts line.split(\,\); if (parts.size() 3) { record.timestamp parts[0].mid(1); // 去除开头引号 record.eventType parts[1]; record.message parts[2].chopped(1); // 去除结尾引号 } return record; }5. 实际应用案例系统日志记录器让我们将这些技术整合到一个实际的系统日志记录器中。5.1 日志记录器设计class SystemLogger { public: SystemLogger() : m_csvHandler(C:/AppLogs) {} void logInfo(const QString message) { logEvent(INFO, message); } void logWarning(const QString message) { logEvent(WARNING, message); } void logError(const QString message) { logEvent(ERROR, message); } QVectorLogEntry getRecentLogs(int maxCount 100) const { return m_csvHandler.readAllRecords().mid(0, maxCount); } private: CSVHandler m_csvHandler; void logEvent(const QString type, const QString message) { if (!m_csvHandler.appendRecord(type, message)) { qCritical() Failed to write log entry to CSV; } } };5.2 多线程日志处理在多线程环境中使用日志记录器// 工作线程示例 void WorkerThread::run() { SystemLogger logger; for (int i 0; i 100; i) { logger.logInfo(QString(Processing item %1).arg(i)); QThread::msleep(100); } } // 主程序中使用 int main(int argc, char *argv[]) { QCoreApplication app(argc, argv); SystemLogger mainLogger; mainLogger.logInfo(Application started); QVectorWorkerThread* threads; for (int i 0; i 5; i) { auto thread new WorkerThread; thread-start(); threads.append(thread); } // ...其他代码... return app.exec(); }5.3 日志分析工具基于CSV日志文件我们可以轻松构建分析工具class LogAnalyzer { public: struct Stats { int infoCount 0; int warningCount 0; int errorCount 0; QDateTime firstEntry; QDateTime lastEntry; }; Stats analyzeLogs(const QString filePath) { Stats stats; CSVHandler handler; auto records handler.readAllRecords(); if (!records.isEmpty()) { stats.firstEntry QDateTime::fromString(records.first().timestamp, yyyy/MM/dd hh:mm:ss); stats.lastEntry QDateTime::fromString(records.last().timestamp, yyyy/MM/dd hh:mm:ss); } for (const auto record : records) { if (record.eventType INFO) stats.infoCount; else if (record.eventType WARNING) stats.warningCount; else if (record.eventType ERROR) stats.errorCount; } return stats; } };

相关文章:

别再手动改Excel了!用QT的QFile和QTextStream搞定CSV读写(附线程安全锁)

用QT实现高效CSV自动化处理:告别Excel手工操作 在数据处理领域,CSV文件因其简单通用而广受欢迎。作为C开发者,我们经常需要处理各种数据导出、日志记录等任务。传统做法可能是手动操作Excel,但这在自动化系统中显然行不通。QT框架…...

深入S32K3芯片内部:图解FCCU状态机与错误处理流程,告别一知半解

深入S32K3芯片内部:图解FCCU状态机与错误处理流程,告别一知半解 在嵌入式系统开发中,功能安全始终是核心考量之一。S32K3系列芯片作为汽车电子和工业控制领域的重要解决方案,其内置的FCCU(Fault Collection and Contro…...

基于安卓的电影评论与观影记录平台毕业设计

博主介绍:✌ 专注于Java,python,✌关注✌私信我✌具体的问题,我会尽力帮助你。一、研究目的本研究旨在设计并实现一个基于安卓操作系统的电影评论与观影记录管理平台以解决当前影视娱乐领域中用户信息交互与个性化服务需求之间的矛盾性问题。随着移动互联…...

【国家级三甲医院实测验证】:基于动态令牌+分片哈希的PHP脱敏新范式(吞吐量提升4.2倍)

更多请点击: https://intelliparadigm.com 第一章:【国家级三甲医院实测验证】:基于动态令牌分片哈希的PHP脱敏新范式(吞吐量提升4.2倍) 在国家卫健委《医疗卫生数据安全管理办法》强制落地背景下,某TOP5…...

【AI 编程工具进阶用法】如何在Cursor、Trae等工具中使用Skills

在Cursor、Trae等非 Claude Code项目使用 Skills,需要用到 OpenSkills 这个开源项目 Skills = 可复用、可调用、可组合的“能力模块”,它是有规则的。 你可以理解为:我让 AI 怎么干活的一整套方法论 + 操作步骤。 https://github.com/numman-ali/openskills openskills介绍…...

校庆(2026)书法展览作品

简 介: 这篇文章记录了校庆书法展的观展体验。作者提到书法作品中篆书、草书等字体识别困难,通过AI工具"豆包"成功识别了多幅作品的文字内容,包括校训、韩愈《马说》、诸葛亮《诫子书》等经典文本。文章展示了十余幅书法作品照片&a…...

Cadence SpectreRF PSS/Pnoise仿真避坑指南:从Beat Frequency设置到Jitter测量实战

Cadence SpectreRF PSS/Pnoise仿真避坑指南:从Beat Frequency设置到Jitter测量实战 在模拟IC设计领域,周期稳态分析(PSS)和相位噪声分析(Pnoise)是评估电路性能的关键工具。然而,许多工程师在实…...

使用 PI 预装载处理永磁同步电机二次起步异响

一、电机二次起步问题 1.1 第一次启动 (电机零速启动) 在电机零速启动时,电机的状态是: (电角速度为 0) (反电动势为 0) 此时我们的控制刚刚启动,给出的其内部的电流环 PI 和输出给定为: 0 (电流环积分器) 0 0 之后积分器和…...

Locust Skill for Claude Code

AI 驱动的高级 API 高并发压测框架 基于 Locust 构建的自动化、配置驱动型压测方案。它集成了接口活性探测、多接口偏移权重管理、AI 性能分析以及自动化的 HTML 报告生成功能。 地址:Locust-Skill 📂 项目结构 (Structure) 为了保持代码库的整洁与可移…...

LeagueAkari:英雄联盟玩家的智能助手与游戏效率提升工具

LeagueAkari:英雄联盟玩家的智能助手与游戏效率提升工具 【免费下载链接】League-Toolkit An all-in-one toolkit for LeagueClient. Gathering power 🚀. 项目地址: https://gitcode.com/gh_mirrors/le/League-Toolkit 还在为英雄联盟中的繁琐操…...

OpenClaw的PDF处理Skill收费吗?

结论:OpenClaw 本身和绝大多数 PDF Skill 都是免费的,只有个别商业引擎有 “免费额度 超额付费” 的模式。一、框架本身(永久免费)OpenClaw 采用 MIT 开源协议,自托管版无订阅费、无隐藏收费。你只需要承担&#xff1…...

SeuratWrappers终极指南:一站式解锁单细胞分析扩展工具集

SeuratWrappers终极指南:一站式解锁单细胞分析扩展工具集 【免费下载链接】seurat-wrappers Community-provided extensions to Seurat 项目地址: https://gitcode.com/gh_mirrors/se/seurat-wrappers 你是否在使用Seurat进行单细胞RNA测序分析时&#xff0c…...

SAP资产折旧对账出现差异时,调整方法取决于差异产生的原因。通常分为两种情况:一是总账多了/少了金额(通常因手工记账导致),二是资产模块多了/少了金额(通常因折旧计算或配置错误导致)

SAP资产折旧对账出现差异时,调整方法取决于差异产生的原因。通常分为两种情况:一是总账多了/少了金额(通常因手工记账导致),二是资产模块多了/少了金额(通常因折旧计算或配置错误导致)。以下是针…...

电信行业语音AI:提升客户体验的技术实践

1. 电信行业客户体验的AI变革:从痛点出发想象一下这样的场景:周一早晨8:15,Jack正在地铁上焦急地拨打电信运营商客服热线,因为他发现自己的5G套餐被错误扣费。电话接通后,他经历了10分钟的等待音乐,期间系统…...

Linux初学——更改SELinux强制模式

目录 SELinux 概述 SELinux 工作模式 SELinux策略——安全规则 常用命令 常见问题排查 实际应用示例 知识点问答题 SELinux 概述 SELinux(Security-Enhanced Linux)是一种基于 Linux 内核的安全模块,通过强制访问控制(MA…...

Layerdivider:如何快速将单张图片智能分层为可编辑PSD文件?

Layerdivider:如何快速将单张图片智能分层为可编辑PSD文件? 【免费下载链接】layerdivider A tool to divide a single illustration into a layered structure. 项目地址: https://gitcode.com/gh_mirrors/la/layerdivider 你是否曾面对一张精美…...

2026 个人站长与开发者云服务器选择指南:性价比 IDC 推荐

2026 个人站长云服务器选择指南:高性价比 IDC 实测推荐 对于个人开发者、学生站长、小程序搭建、测试环境、小型网站来说,选择一台稳定、便宜、灵活的云服务器至关重要。大厂云虽然稳定,但价格偏高;小厂商良莠不齐,很容…...

从仿真到电路设计:如何将Lumerical FDTD的环形谐振器S参数导入INTERCONNECT进行系统级分析

光子集成电路设计实战:从FDTD仿真到INTERCONNECT系统级分析的完整工作流 硅基光子集成电路(PIC)设计正经历从实验室研究到商业化量产的关键转型期。全球领先的代工厂已开始提供标准化硅光工艺设计套件(PDK)&#xff0c…...

2025_NIPS_Train Once, Get a Family: State-Adaptive Balances for Offline-to-Online Reinforcement L...

文章核心总结与翻译 一、主要内容 该研究聚焦于离线到在线强化学习(Offline-to-Online RL) 范式,解决现有算法在处理分布偏移问题时采用"一刀切"的策略改进与约束平衡方式导致的性能瓶颈。核心内容包括: 问题背景:离线到在线RL通过离线预训练(利用已有数据集…...

算法训练营第十八天|20. 有效的括号

题目链接:https://leetcode.cn/problems/valid-parentheses/ 视频讲解:https://www.bilibili.com/video/BV1AF411w78g 代码实现: 1、看到题目的第一想法: 看到题目,我立刻想到用栈来解决:遇到左括号就入栈…...

算法训练营第十七天|151.翻转字符串里的单词

题目链接:https://leetcode.cn/problems/reverse-words-in-a-string/ 视频讲解:https://www.bilibili.com/video/BV1uT41177fX 代码实现: 1、看到题目的第一想法: 看到题目,我首先想到:先去除多余空格&am…...

特征学习电力机车辅助供电接地故障诊断【附代码】

✅ 博主简介:擅长数据搜集与处理、建模仿真、程序设计、仿真代码、论文写作与指导,毕业论文、期刊论文经验交流。 ✅ 如需沟通交流,扫描文章底部二维码。(1)故障机理驱动的时序波形特征建模:针对电力机车辅…...

实时说话人识别技术:Streaming Sortformer解析与应用

1. 实时说话人识别技术的现状与挑战在多人语音交互场景中,准确识别"谁在什么时候说话"一直是个技术难题。传统方案要么依赖专用硬件阵列麦克风,要么只能通过事后批处理实现,无法满足实时性要求。我曾参与过多个语音识别项目&#x…...

户外LED照明电路保护设计与器件选型指南

1. 户外LED照明的发展现状与技术优势LED照明技术在过去十年中经历了爆发式增长,这主要得益于其显著的节能特性和超长使用寿命。与传统白炽灯相比,LED的光效(每瓦电力产生的光通量)高出5-8倍,这使得全球各国政府纷纷出台…...

AI Agent不是未来,是现在——程序员该如何抓住这波红利

先说结论 2026年,不会用AI Agent的程序员,就像2008年不会用Google的程序员。 这不是危言耸听。腾讯、阿里、字节都在All in这个方向,OpenClaw/Agent的爆火只是开始。 今天这篇文章,我会用"冰箱理论"帮你彻底搞懂AI Agent,再告诉你怎么快速上车。 这个东西是什…...

告别格式限制:用ncmdump解锁网易云音乐NCM加密文件,实现跨平台自由播放

告别格式限制:用ncmdump解锁网易云音乐NCM加密文件,实现跨平台自由播放 【免费下载链接】ncmdump 项目地址: https://gitcode.com/gh_mirrors/ncmd/ncmdump 当您从网易云音乐下载的NCM加密文件无法在车载音响、手机自带播放器或其他音乐软件中播…...

本地部署AI对话伴侣:基于Ollama与角色提示词的私有化ChatGPT实践

1. 项目概述:一个开源的本地AI对话伴侣如果你对AI聊天机器人感兴趣,但又对数据隐私、网络延迟或者高昂的API费用心存顾虑,那么你很可能已经听说过“本地部署”这个概念。今天要聊的这个项目,withcatai/catai,就是一个将…...

OpCore Simplify:让黑苹果配置从复杂到简单的智能助手

OpCore Simplify:让黑苹果配置从复杂到简单的智能助手 【免费下载链接】OpCore-Simplify A tool designed to simplify the creation of OpenCore EFI 项目地址: https://gitcode.com/GitHub_Trending/op/OpCore-Simplify OpCore Simplify是一款专为黑苹果&a…...

从零搭建 K8s 集群 + Prometheus 监控 + Harbor 私有仓库 + 钉钉告警(RHEL 9 实战)

记录一下最近在 RHEL 9 环境下从零搭建 K8s 集群、Prometheus 监控体系、Harbor 私有镜像仓库的完整过程,踩了不少坑,全部记录下来。 环境说明 主机名IP角色系统Ubuntu192.168.137.128Prometheus Grafana AlertmanagerUbuntu 22.04k8s-master192.168…...

Kubernetes智能运维实践:基于大语言模型的AI副驾驶工具详解

1. 项目概述:当Kubernetes遇上AI副驾驶如果你和我一样,每天都要和成百上千个Kubernetes Pod、Service、Ingress打交道,那一定经历过这样的时刻:凌晨三点被告警叫醒,面对一个不断重启的Pod,日志刷屏却找不到…...