实现QT中qDebug()的日志重定向

背景:

在项目开发过程中,为了方便分析和排查问题,我们需要将原本输出到控制台的调试信息写入日志文件,进行持久化存储,还可以实现日志分级等。

日志输出格式:

在这里插入图片描述
我们需要的格式包括以下内容:
1.[2024-02-19 10:21:11:387]: 时间戳,表示日志记录的时间,包括年、月、日、时、分、秒、毫秒。
2.thread:0x7f44aa137600: 线程信息,表示记录日志的线程ID。
3.[Debug ]: 日志级别,这里是Debug级别,表示是调试信息。
4.[sampleapply.cpp line:972 void SampleApply::on_applyBtn_clicked]: 文件名、行号和函数名信息,指明日志记录所在的源文件、行号和函数。
5.{ 打印日志}: 具体的日志内容,这里是一个占位符,实际应该是打印的具体日志信息。

按照以上信息,接下来我会说明如何实现这些内容的打印输出以及qDebug()的重定向。

一、根据上下文提取文件名、函数名

static QString getFileName(QString logContext)
{if (logContext.isEmpty())return QString("");//输入是相对路径名,只取其中的文件名/*** windows格式:"..\\UI\\src\\rpc\\uimessage.cpp"* ubuntu格式: "../UI/src/rpc/uimessage.cpp"*/QString fileName(logContext);int start = 0;
#ifdef Q_OS_WIN32start = fileName.lastIndexOf("\\")+1;
#elsestart = fileName.lastIndexOf("/")+1;
#endifreturn fileName.mid(start);
}说明:
根据给定的源文件路径 logContext 提取并返回文件名。该函数的主要作用是根据不同操作系统的路径格式,提取文件名,从而满足跨平台开发。
static QString getFunName(QString logContext)
{if (logContext.isEmpty())return QString("");//有两个情况,只取其中的函数名,ubuntu下该格式不适用// void __cdecl MainWindow::onBootup()// int __cdecl main(int,char *[])QString funName(logContext);int start = 0;
#ifdef Q_OS_WIN32		//使用预处理器指令,判断当前编译环境是否为 Windowsstart = funName.lastIndexOf("decl")+4;
#endifint end   = funName.indexOf("(");return funName.mid(start, end-start);
}说明:
根据给定的日志上下文 logContext 提取并返回函数名。该函数的主要作用是解析函数名的不同形式(在 Windows 环境下,一些函数可能带有 __cdecl 标识)。

二、处理日志的输出

void LogManager::outputLog(QtMsgType type, const QMessageLogContext &context, const QString &msg)
{//如果要写文件需要加锁,以确保在多线程环境中正确处理日志输出QMutexLocker locker(&mLogMutex);QString out_text;QTextStream stream(&out_text);//打印时间和线程ID信息stream<<QDateTime::currentDateTime().toString("[yyyy-MM-dd hh:mm:ss:zzz] ") << "thread:" << QThread::currentThreadId() << " ";//根据日志类型在字符串头部加上相应的标识,如 "[Debug]", "[Info]", "[Warning]", "[Critical]", "[Fatal]"等switch (type) {case QtDebugMsg:        stream<<"[Debug   ]"; break;case QtInfoMsg:         stream<<"[Info    ]"; break;case QtWarningMsg:      stream<<"[Warning ]"; break;case QtCriticalMsg:     stream<<"[Critical]"; break;case QtFatalMsg:        stream<<"[Fatal   ]"; break;default:                stream<<"[Unknown ]"; break;}//调用 getFileName 和 getFunName 函数获取文件名和函数名,并添加到日志中QString fileName = getFileName(context.file);       ///<打印所在的源文件名称QString funName = getFunName(context.function);     ///<函数名称QString tmpStr = QString("[%1 line:%2 %3] { %4}").arg(fileName).arg(context.line).arg(funName).arg(msg);stream << tmpStr;//日志大小分割,分割大小默认为5MB,此功能默认不开启if(mIsFileSplit){QFileInfo info(mFile.fileName());//获取文件大小并进行比较if(info.size() >= mFileSize*1024*1024){//超过设定值后重命名该文件,关闭文件mFile.rename(mFilePath + QDate::currentDate().toString("yyyyMMdd")+"_ui_" + QString::number(LOG_INDEX) + ".txt");if(mFile.isOpen()){mFile.close();}//建立并打开新的文件准备进行写入mFile.setFileName(mFilePath + QDate::currentDate().toString("yyyyMMdd")+"_ui_" + QString::number(++LOG_INDEX) + ".txt");if(!mFile.open(QIODevice::WriteOnly|QIODevice::Append)){emit newLog(QtDebugMsg,"Open log file error:"+mFile.errorString()+mFile.fileName());        //打开失败发送错误消息}}}if(!mFile.isOpen()){//文件未打开则按照日期设定文件的名称mFile.setFileName(mFilePath + QDate::currentDate().toString("yyyyMMdd")+"_ui" + ".log");//打开文件,使用Append追加模式,避免同一文件被清除if(!mFile.open(QIODevice::WriteOnly|QIODevice::Append)){emit newLog(QtDebugMsg,"Open log file error:"+mFile.errorString()+mFile.fileName());        //打开失败发送错误消息}}//文件打开则直接进行写入if(mFile.isOpen()){//写入文件stream.setDevice(&mFile);stream<<out_text<<Qt::endl;}//发送信号给需要的对象,如ui上显示日志emit newLog(type, msg);//默认的输出,控制台//区分日志类型给文本加颜色//常见格式为:\e[显示方式;前景色;背景色m输出字符串\e[0m//其中\e=\033QString cmd_text;stream.setString(&cmd_text);switch (type) {//debug绿色case QtDebugMsg:        stream<<"\033[34m"; break;//info蓝色case QtInfoMsg:         stream<<"\033[34m"; break;//warning紫色case QtWarningMsg:      stream<<"\033[35m"; break;//critical加粗红色case QtCriticalMsg:     stream<<"\033[1;31m"; break;//fatal红底黑字//Fatal表示致命错误,默认处理会报异常的case QtFatalMsg:        stream<<"\033[1;30;41m"; break;//defualt默认颜色default:                stream<<"\033[0m"; break;}stream<<out_text<<"\033[0m";mDefaultOutput(type,context,cmd_text);
}

三、日志的重定向

在程序运行时将 Qt 的调试输出(qDebug() 等)重定向到自定义的日志处理函数,以实现对日志的自定义记录和处理。

//重定向qdebug输出
void outputLogMessage(QtMsgType type, const QMessageLogContext& context, const QString& msg)
{//转发给单例的成员函数LogManager::getInstance()->outputLog(type,context,msg);
}//在构造函数中调用:
void LogManager::initManager(const QString &path)
{//保存路径mFilePath=path;if(mFilePath.isEmpty()){//使用QDir直接获取当前路径mFilePath = QDir::currentPath()+"/log/ui/";}QDir dir(mFilePath);if(!dir.exists()){dir.mkpath(mFilePath);}//重定向qdebug到自定义函数mDefaultOutput=qInstallMessageHandler(outputLogMessage);
}

四、释放资源

在释放 LogManager 类时,我们需要确保相关资源的正确释放,包括关闭已打开的日志文件,并取消对消息的自定义处理。

LogManager::LogManager():mFileSize(5),mIsFileSplit(false)
{initManager();
}void LogManager::freeManager()
{mFile.close();qInstallMessageHandler(nullptr);
}

以上就是本文全部内容,欢迎一起讨论!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQL死锁详细介绍

首先死锁产生的原因&#xff1a;两个及以上事务争夺资源导致互相等待造成的 比如事务A先修改id为1的数据再去修改id为2的数据&#xff0c;事务B先修改id为2的数据再去修改id为1的数据&#xff0c;因为事务A先拿到id1的锁再去拿id2的锁&#xff0c;而事务B先拿到id2的锁又去拿id…

“成像光谱遥感技术中的AI革命:ChatGPT应用指

遥感技术主要通过卫星和飞机从远处观察和测量我们的环境&#xff0c;是理解和监测地球物理、化学和生物系统的基石。ChatGPT是由OpenAI开发的最先进的语言模型&#xff0c;在理解和生成人类语言方面表现出了非凡的能力。本文重点介绍ChatGPT在遥感中的应用&#xff0c;人工智能…

【Algorithms 4】算法(第4版)学习笔记 16 - 4.2 有向图

文章目录 前言参考目录学习笔记1&#xff1a;介绍1.1&#xff1a;有向图简介1.2&#xff1a;应用举例1.3&#xff1a;相关问题2&#xff1a;有向图 API2.1&#xff1a;有向图表示2.1.1&#xff1a;邻接表数组 Adjacency-list2.1.2&#xff1a;Java 实现&#xff1a;邻接表数组2…

2. gin中间件注意事项、路由拆分与注册技巧

文章目录 一、中间件二、Gin路由简介1、普通路由2、路由组 三、路由拆分与注册1、基本的路由注册2、路由拆分成单独文件或包3、路由拆分成多个文件4、路由拆分到不同的APP 一、中间件 在日常工作中&#xff0c;经常会有一些计算接口耗时和限流的操作&#xff0c;如果每写一个接…

Sftp服务器搭建(linux)

Sftp服务器搭建&#xff08;linux&#xff09; 一、基本工作原理 FTP的基本工作原理如下&#xff1a; 1&#xff09;建立连接&#xff1a;客户端与服务器之间通过TCP/IP建立连接。默认情况下&#xff0c;FTP使用端口号21作为控制连接的端口。​​​​​​​ 2&#xff09;身…

基于AI软件平台 HEGERLS智能托盘四向车机器人物流仓储解决方案持续升级

随着各大中小型企业对仓储需求的日趋复杂&#xff0c;柔性、离散的物流子系统也不断涌现&#xff0c;各种多类型的智能移动机器人、自动化仓储装备大量陆续的应用于物流行业中&#xff0c;但仅仅依靠传统的物流技术和单点的智能化设备&#xff0c;已经无法更有效的应对这些挑战…

Office 2007软件安装教程(附软件下载地址)

软件简介&#xff1a; 软件【下载地址】获取方式见文末。注&#xff1a;推荐使用&#xff0c;更贴合此安装方法&#xff01; 微软Office 2007是一款具有重大创新与革命性的办公软件套件。它引入了全新设计的用户界面&#xff0c;提供稳定安全的文件格式&#xff0c;并实现了无…

数据结构 - 堆(优先队列)+ 堆的应用 + 堆练习

文章目录 前言堆一、什么是堆二、堆又分为大根堆和小根堆三、由于堆的逻辑结构被看成完全二叉树&#xff0c;那么我们先来了解一下完全二叉树。四、堆使用数组还是链表储存数据呢&#xff1f;五、数组构建二叉树和父子节点之间的定位六、堆进行的操作七、实现小根堆1、堆的初始…

算法(数据结构)面试问题准备 二分法/DFS/BFS/快排

一、算法概念题 1. 二分法 总结链接几种查找情况的模板另一个好记的总结总结&#xff1a;搜索元素两端闭&#xff0c;while带等&#xff0c;mid1&#xff0c;结束返-1 搜索边界常常左闭右开&#xff0c;while小于&#xff0c;mid看边界开闭&#xff0c;闭开&#xff0c;结束i…

vue2【详解】生命周期(含父子组件的生命周期顺序)

1——beforeCreate&#xff1a;在内存中创建出vue实例&#xff0c;数据观测 (data observer) 和 event/watcher 事件配置还没调用&#xff08;data 和 methods 属性还没初始化&#xff09; 【执行数据观测 (data observer) 和 event/watcher 事件配置】 2——created&#xf…

指纹加密U盘/指纹KEY方案——采用金融级安全芯片 ACH512

方案概述 指纹加密U盘解决方案可实现指纹算法处理、数据安全加密、数据高速存取&#xff08;EMMC/TF卡/NandFlash&#xff09;&#xff0c;可有效保护用户数据安全。 方案特点 • 采用金融级安全芯片 ACH512 • 存储介质&#xff1a;EMMC、TF卡、NandFlash • 支持全系列国密…

解决白屏问题:让你的网站重焕生机

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

[Flutter]使用Provider进行状态管理

一、使用Provider进行状态管理的基本用法 Provider是Flutter中一个非常流行的状态管理工具&#xff0c;它可以帮助开发者更有效地管理Widget树中的数据。Provider的核心思想是将数据模型放置在Widget树中可以被多个子Widget访问的地方&#xff0c;而不必通过构造函数手动传递。…

软件测试 —— 如何测试图片上传功能?

作为一名专业的软件测试人员&#xff0c;测试图片上传功能是一个重要的任务&#xff0c;以下是一些测试该功能的常用方法&#xff1a; 1. 上传功能测试&#xff1a;确保图片上传功能正常工作&#xff0c;包括选择图片文件、点击上传按钮、上传进度显示、上传成功/失败的提示等。…

JavaWeb03-HTTP协议,Tomcat,Servlet

目录 一、HTTP协议 1.概述 2.特点 3.请求数据格式 &#xff08;1&#xff09;请求行 &#xff08;2&#xff09;请求头 &#xff08;3&#xff09;请求体 &#xff08;4&#xff09;常见请求头 &#xff08;5&#xff09;GET和POST请求区别 4.响应数据格式 &#xf…

gRPC-第二代rpc服务

在如今云原生技术的大环境下&#xff0c;rpc服务作为最重要的互联网技术&#xff0c;蓬勃发展&#xff0c;诞生了许多知名基于rpc协议的框架&#xff0c;其中就有本文的主角gRPC技术。 一款高性能、开源的通用rpc框架 作者作为一名在JD实习的Cpper&#xff0c;经过一段时间的学…

Flask python开发篇: 写一个简单的接口

第一步&#xff1a;新建flask项目 参考使用pycharm新建一个项目 打开pycharm&#xff0c;根据下面图中箭头顺序&#xff0c;新建一个flask的项目&#xff1b; 第二步&#xff1a;运行项目&#xff0c; 安装成功以后&#xff0c;会有个app.py文件&#xff0c;打开以后&#…

qt一个项目有且只有有一个maindow,其他小窗口用QWidget,QDialog是带有yes和no的QWidget

QMaindow QWidget QDialog区别很大 我想要在生成一个小窗口&#xff0c;结果选择基类为maindow&#xff0c;应该是QWidget 然后就出现奇奇怪怪的问题 QMaindow和QWidget不能乱选择&#xff0c;而且各自QPaintEvent也有很多区别 以下就是错误&#xff1a; 我继承maindow的基类…

【物联网设备端开发】物联网设备上云提供开箱即用接入SDK

&#x1f308; 个人主页&#xff1a;帐篷Li &#x1f525; 系列专栏&#xff1a;物联网设备端开发 &#x1f4aa;&#x1f3fb; 物联网设备上云提供开箱即用接入SDK 目录 一、项目介绍 二、项目目录 三、集成方式 四、设备功能开发 4.1 连接与消息 4.2 业务功能 4.3 运维…

云服务器实例重启后,各个微服务的接口(涉及mysql操作的)都用不了了

问题描述&#xff1a; 云服务器被黑客植入挖矿。重启云服务器实例后得到解决&#xff0c;接着把docker&#xff08;zookeeper、redis啥的&#xff09;还有后端jar包啥的都重启了&#xff0c;然后发现后端接口访问不了&#xff0c;只有不涉及数据库操作的接口正常访问&#xff…