Qt实现QDebug重定向输出到日志文件(支持多线程安全)

2024-04-02 11:52

本文主要是介绍Qt实现QDebug重定向输出到日志文件(支持多线程安全),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在Qt中,qDebug(),qInfo(),qWarning(),qCritical(),qFatal()常用于打印信息到终端控制台,我们可以将其重定向输出到文件中。

我们创建一个单例类“LogOutput”封装重定向到文件操作:

logoutput.h

#ifndef LOGOUTPUT_H
#define LOGOUTPUT_H
/*** @brief QDebug的日志输出重定向到文件* @author wjp* @date 2024.3.8 modify by wjp** */#include <QObject>
#include <QtMessageHandler>
#include <QDebug>
#include <QFile>
#include <QMutex>
#include <QMutexLocker>#define LOG_MAXSIZE  5 * 1024 * 1024 //单个log文件最大值class LogOutput : public QObject
{Q_OBJECT
public:static LogOutput * getInstance();void install(); //安装信息处理函数void uninstall(); //卸载信息处理函数void deleteLog(); //删除过期日志protected:// 此函数用于注册static void customMessageHandler(QtMsgType type, const QMessageLogContext &context, const QString &msg);private://将构造函数,拷贝构造,赋值运算符都定义为私有,不允许外部类操作explicit LogOutput(QObject *parent = nullptr); //构造函数~LogOutput();  //析构LogOutput(const LogOutput &sig) = delete; //拷贝构造函数LogOutput& operator=(const LogOutput &sig) = delete ; //赋值运算符重载private:static LogOutput *ins;  //私有静态对象static QMutex m_mutex;QFile m_curLogFile;QString m_curLogFileDate;// 当前日志所属日期,防止在午夜24点刚刚过,需要更换日志输出文件//信息处理函数(重写的myMessageHandler)/*功能说明:通过调试信息保存到日志文件**参数说明:* msgType: 调试信息类型或级别(qdebug, qwarning, qfatal 。。。。)* context: 调试信息所处文本,可使用context.file和context.line获取文本所处行数及所处文件路径,以及使用context.function获取文本所处函数名* msg: 调试信息内容,自定义*/void outPutMsg(QtMsgType msgType, const QMessageLogContext &context, const QString &msg);/** 函数功能:* 1、根据调试信息以及日期,保存到相应的文件。* 2、在保存文件前需要判断文件大小是否大于自定义值,如果大于,便按照序号从小到大新建一个。*/void saveLog(QString message);// 打开日志文件void openTheLogFile();
};#endif // LOGOUTPUT_H

logoutput.cpp(注意输出日志文件的路径你可以优化一下,通过公共接口设置)

#include "logoutput.h"
#include <QFile>
#include <QTextStream>
#include <QApplication>
#include <QDateTime>
#include <QMutex>
#include <QFileInfo>
#include <QMetaEnum>
#include <QSettings>
#include <QDir>LogOutput * LogOutput::ins = nullptr;
QMutex LogOutput::m_mutex;LogOutput::LogOutput(QObject *parent) : QObject(parent)
{}LogOutput::~LogOutput()
{if(m_curLogFile.isOpen())m_curLogFile.close();qInfo() << "日志输出模块释放";
}//安装日志函数
void LogOutput::install()
{//创建log文件夹QString logPath = QApplication::applicationDirPath() + "/log";QDir dir(logPath);if(!dir.exists()){dir.mkdir(logPath);qDebug() << QString("运行日志文件夹创建成功:%1").arg(logPath);}// 打开日志输出文件(不存在则创建并打开)openTheLogFile();//安装消息处理函数qInstallMessageHandler(LogOutput::customMessageHandler);// 此句执行后,qDebug,qInfo等才会输出到文件qInfo() << "-------------------日志输出模块创建成功-------------------";//deleteLog(); //删除过期日志
}//卸载日志函数
void LogOutput::uninstall()
{qInstallMessageHandler(nullptr);
}//日志信息处理函数
void LogOutput::outPutMsg(QtMsgType msgType, const QMessageLogContext &context, const QString &msg)
{//判断信息类型QString type;switch (msgType){case QtDebugMsg:type = QString("Debug");break;case QtWarningMsg:type = QString("Warning");break;case QtCriticalMsg:type = QString("Critical");break;case QtFatalMsg:type = QString("Fatal");break;case QtInfoMsg:type = QString("Info");}m_mutex.lock();  //互斥关锁//文件名和行数以及函数QString contextInfo = QString("[File:(%1), Line:(%2), Funtion(%3)]:").arg(context.file).arg(context.line).arg(context.function);//获取当前时间,精确到秒QDateTime curdatetime = QDateTime::currentDateTime();QString currentDate = curdatetime.date().toString("yyyy-MM-dd");QString currentTime = curdatetime.time().toString("hh:mm:ss");if(m_curLogFile.isOpen()){bool bFileSizeLarge = (m_curLogFile.size() >= LOG_MAXSIZE) ? true : false;bool bNextDate = (currentDate.compare(m_curLogFileDate,Qt::CaseInsensitive) == 0) ? false : true;// 当系统时间过了今天的24点到达另一天,或者超过单个日志文件最大值,切换日志输出文件if(bFileSizeLarge || bNextDate){m_curLogFile.close();// 重新打开日志输出文件(不存在则创建并打开)openTheLogFile();}}//拼接信息字符串QString message = QString("[%1 %2] %3: %4 %5").arg(currentDate).arg(currentTime).arg(type).arg(contextInfo).arg(msg);//存入信息到日志文件saveLog(message);m_mutex.unlock(); //开锁}//删除过期日志
void LogOutput::deleteLog()
{//获取日志文件夹地址QString dirName =  QApplication::applicationDirPath() + "/log";QDir dir(dirName);//获取文件夹下所有文件信息列表QFileInfoList infoList = dir.entryInfoList(QDir::Files);//遍历日志文件foreach (QFileInfo fileInfo, infoList) {//将文件创建时间与过期时间作比较,如果创建时间小于过期时间,则删除(代码是一天期限,如果改月为单位可以使用addMonths)if(fileInfo.birthTime() <= QDateTime::currentDateTime().addDays(-1)){QFile::setPermissions(dirName + "/" +fileInfo.fileName(), QFileDevice::ReadOther | QFileDevice::WriteOther);if(QFile::remove(dirName + "/" +fileInfo.fileName())){qDebug() << "过期日志文件删除成功!-->" << fileInfo.fileName();}else{qDebug() << "过期日志文件删除成功!-->" << fileInfo.fileName();}}}
}//保存日志到文件
void LogOutput::saveLog(QString message)
{if(m_curLogFile.isOpen()){QTextStream write(&m_curLogFile);write << message << "\r\n";m_curLogFile.flush();// 刷新写入缓存}
}void LogOutput::openTheLogFile()
{int i = 1; //当文件大小超过最大值时,给新文件添加编号m_curLogFileDate = QDateTime::currentDateTime().toString("yyyy-MM-dd");//以天为单位给文件命名QString fileName = QApplication::applicationDirPath() + "/log/" + m_curLogFileDate + "_log";//文件名右边(后缀)QString fileNameRight;//最终要写入的文件名QString fileNameLast = fileName + ".txt";//绑定文件对象m_curLogFile.setFileName(fileNameLast);//判断文件大小while(m_curLogFile.size() >= LOG_MAXSIZE){//给新文件加入序号后缀fileNameRight = QString("_%1.txt").arg(i);//拼接最终文件名fileNameLast = fileName + fileNameRight;//修改file绑定的文件名m_curLogFile.setFileName(fileNameLast);i++;}//只写和拼接的方式打开文件bool isopen = m_curLogFile.open(QIODevice::WriteOnly | QIODevice::Append);if(isopen == true){}
}LogOutput *LogOutput::getInstance()
{QMutexLocker locker(&m_mutex);  //必须加锁if(!ins){ins = new LogOutput();}return ins;
}void LogOutput::customMessageHandler(QtMsgType type, const QMessageLogContext &context, const QString &msg)
{LogOutput::getInstance()->outPutMsg(type, context, msg); // 调用非静态成员函数处理消息
};

main.cpp中使用

int main(int argc, char *argv[])
{QApplication app(argc, argv);// 日志模块初始化(将QDebug重定向到文件)LogOutput::getInstance()->install();// 测试重定向后是否有这些日志输出到文件qDebug() << "日志输出1";qInfo() << "日志输出2";qWarning() << "日志输出3";qCritical() << "日志输出4";qFatal() << "日志输出5";MainWindow w;w.show();return app.exec();
}

执行LogOutput::getInstance()->uninstall()可以取消重定向恢复打印到终端控制台。

这篇关于Qt实现QDebug重定向输出到日志文件(支持多线程安全)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

MySQL中查找重复值的实现

《MySQL中查找重复值的实现》查找重复值是一项常见需求,比如在数据清理、数据分析、数据质量检查等场景下,我们常常需要找出表中某列或多列的重复值,具有一定的参考价值,感兴趣的可以了解一下... 目录技术背景实现步骤方法一:使用GROUP BY和HAVING子句方法二:仅返回重复值方法三:返回完整记录方法四:

IDEA中新建/切换Git分支的实现步骤

《IDEA中新建/切换Git分支的实现步骤》本文主要介绍了IDEA中新建/切换Git分支的实现步骤,通过菜单创建新分支并选择是否切换,创建后在Git详情或右键Checkout中切换分支,感兴趣的可以了... 前提:项目已被Git托管1、点击上方栏Git->NewBrancjsh...2、输入新的分支的

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Python实现对阿里云OSS对象存储的操作详解

《Python实现对阿里云OSS对象存储的操作详解》这篇文章主要为大家详细介绍了Python实现对阿里云OSS对象存储的操作相关知识,包括连接,上传,下载,列举等功能,感兴趣的小伙伴可以了解下... 目录一、直接使用代码二、详细使用1. 环境准备2. 初始化配置3. bucket配置创建4. 文件上传到os

解读GC日志中的各项指标用法

《解读GC日志中的各项指标用法》:本文主要介绍GC日志中的各项指标用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、基础 GC 日志格式(以 G1 为例)1. Minor GC 日志2. Full GC 日志二、关键指标解析1. GC 类型与触发原因2. 堆

Qt QCustomPlot库简介(最新推荐)

《QtQCustomPlot库简介(最新推荐)》QCustomPlot是一款基于Qt的高性能C++绘图库,专为二维数据可视化设计,它具有轻量级、实时处理百万级数据和多图层支持等特点,适用于科学计算、... 目录核心特性概览核心组件解析1.绘图核心 (QCustomPlot类)2.数据容器 (QCPDataC

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

关于集合与数组转换实现方法

《关于集合与数组转换实现方法》:本文主要介绍关于集合与数组转换实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、Arrays.asList()1.1、方法作用1.2、内部实现1.3、修改元素的影响1.4、注意事项2、list.toArray()2.1、方

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

java实现docker镜像上传到harbor仓库的方式

《java实现docker镜像上传到harbor仓库的方式》:本文主要介绍java实现docker镜像上传到harbor仓库的方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 前 言2. 编写工具类2.1 引入依赖包2.2 使用当前服务器的docker环境推送镜像2.2