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

2024-03-09 06:44

本文主要是介绍实现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);
}

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

这篇关于实现QT中qDebug()的日志重定向的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring Boot集成/输出/日志级别控制/持久化开发实践

《SpringBoot集成/输出/日志级别控制/持久化开发实践》SpringBoot默认集成Logback,支持灵活日志级别配置(INFO/DEBUG等),输出包含时间戳、级别、类名等信息,并可通过... 目录一、日志概述1.1、Spring Boot日志简介1.2、日志框架与默认配置1.3、日志的核心作用

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

Redis客户端连接机制的实现方案

《Redis客户端连接机制的实现方案》本文主要介绍了Redis客户端连接机制的实现方案,包括事件驱动模型、非阻塞I/O处理、连接池应用及配置优化,具有一定的参考价值,感兴趣的可以了解一下... 目录1. Redis连接模型概述2. 连接建立过程详解2.1 连php接初始化流程2.2 关键配置参数3. 最大连

Python实现网格交易策略的过程

《Python实现网格交易策略的过程》本文讲解Python网格交易策略,利用ccxt获取加密货币数据及backtrader回测,通过设定网格节点,低买高卖获利,适合震荡行情,下面跟我一起看看我们的第一... 网格交易是一种经典的量化交易策略,其核心思想是在价格上下预设多个“网格”,当价格触发特定网格时执行买

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统

深度解析Nginx日志分析与499状态码问题解决

《深度解析Nginx日志分析与499状态码问题解决》在Web服务器运维和性能优化过程中,Nginx日志是排查问题的重要依据,本文将围绕Nginx日志分析、499状态码的成因、排查方法及解决方案展开讨论... 目录前言1. Nginx日志基础1.1 Nginx日志存放位置1.2 Nginx日志格式2. 499

Python对接支付宝支付之使用AliPay实现的详细操作指南

《Python对接支付宝支付之使用AliPay实现的详细操作指南》支付宝没有提供PythonSDK,但是强大的github就有提供python-alipay-sdk,封装里很多复杂操作,使用这个我们就... 目录一、引言二、准备工作2.1 支付宝开放平台入驻与应用创建2.2 密钥生成与配置2.3 安装ali

Spring Security 单点登录与自动登录机制的实现原理

《SpringSecurity单点登录与自动登录机制的实现原理》本文探讨SpringSecurity实现单点登录(SSO)与自动登录机制,涵盖JWT跨系统认证、RememberMe持久化Token... 目录一、核心概念解析1.1 单点登录(SSO)1.2 自动登录(Remember Me)二、代码分析三、

PyCharm中配置PyQt的实现步骤

《PyCharm中配置PyQt的实现步骤》PyCharm是JetBrains推出的一款强大的PythonIDE,结合PyQt可以进行pythion高效开发桌面GUI应用程序,本文就来介绍一下PyCha... 目录1. 安装China编程PyQt1.PyQt 核心组件2. 基础 PyQt 应用程序结构3. 使用 Q

Python实现批量提取BLF文件时间戳

《Python实现批量提取BLF文件时间戳》BLF(BinaryLoggingFormat)作为Vector公司推出的CAN总线数据记录格式,被广泛用于存储车辆通信数据,本文将使用Python轻松提取... 目录一、为什么需要批量处理 BLF 文件二、核心代码解析:从文件遍历到数据导出1. 环境准备与依赖库