Qt5实现qDebug日志信息写入日志文件过程

 更新时间:2022年05月09日 15:16:38   作者:xiaoyaoyou.xyz  
这篇文章主要为大家介绍了Qt5实现qDebug日志信息写入日志文件的过程示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

1、思路

其实Qt提供了将日志写入文件的方式,是以接口的形式提供的,我们只需要按照我们的需求完成回调函数和我们自己的接口即可。

为方便调用,我们一般直接写到log.h中即可。

  • 1、创建log.h
  • 2、根据我们自己的需求完成回调函数
  • 3、根据需求封装外部调用函数并注册回调函数

2、代码

我这里按照我的需求,当接口传递日志文件名和等级时使用传递的文件存储日志,否则以精确到当前时间为文件名创建日志文件。

提供给外部的调用函数包含日志文件名和日志等级两个接口,可以不传递,直接使用默认值。

Log.h

#ifndef LOG_H#define LOG_H#include <QFile>#include <QTextStream>#include <QDateTime>#include <QMutex>//选择屏幕打印还是输出到文件可以根据这个宏控制或者控制函数调用位置都可以//#define _DEBUG//默认调试级别为warning,即小于warning级别的都不会写入日志文件//只有release版本的时候,才会输出到日志,debug版本正常输出到终端。namespace QT_LOG{<!--{C}%3C!%2D%2D%20%2D%2D%3E--> //默认文件名为当前时间命名的log文件 static int m_LogLevel = 1; static QString m_LogFile = QString("%1.log").arg(QDateTime::currentDateTime().toString("yyyyMMddhhmmss")); QMutex m_LogMutex; void customMessageHandler(QtMsgType type , const QMessageLogContext &context , const QString &msg) {<!--{C}%3C!%2D%2D%20%2D%2D%3E--> //设置输出日志级别,小于该级别,将不会写入日志文件,默认是warning级别,即debug信息不会写入日志文件 if (type < m_LogLevel) {<!--{C}%3C!%2D%2D%20%2D%2D%3E--> return; } QString log_info; switch (type) {<!--{C}%3C!%2D%2D%20%2D%2D%3E--> case QtDebugMsg: log_info = QString("%1[Debug]:").arg(QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss")); break; case QtWarningMsg: log_info = QString("%1[Warning]:").arg(QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss")); break; case QtCriticalMsg: log_info = QString("%1[Critical]:").arg(QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss")); break; case QtFatalMsg: log_info = QString("%1[Fatal]:").arg(QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss")); abort(); case QtInfoMsg: log_info = QString("%1[Info]:").arg(QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss")); break; } log_info += QString(context.file) + QString(context.line) + QString("%1").arg(msg); //为了线程安全 m_LogMutex.lock(); QFile outFile(m_LogFile); outFile.open(QIODevice::WriteOnly | QIODevice::Append | QIODevice::Text); QTextStream ts(&outFile); ts << log_info << endl; outFile.close(); m_LogMutex.unlock(); } //默认调试级别为warning及以上才会写入日志文件,默认log文件名为程序启动时间命名的log文件 void logInit(QString logFile = "", int logLevel = 1) {<!--{C}%3C!%2D%2D%20%2D%2D%3E--> #ifndef _DEBUG //实现debug版本的时候,输出到终端;release版本的时候输出到日志文件 if ((logLevel < 0) || (logLevel > 3)) {<!--{C}%3C!%2D%2D%20%2D%2D%3E--> m_LogLevel = 1; } else {<!--{C}%3C!%2D%2D%20%2D%2D%3E--> m_LogLevel = logLevel; } if (!logFile.isEmpty()) {<!--{C}%3C!%2D%2D%20%2D%2D%3E--> m_LogFile = logFile; } qInstallMessageHandler(customMessageHandler); #endif }}#endif // LOG_H#ifndef LOG_H
#define LOG_H
#include <QFile>
#include <QTextStream>
#include <QDateTime>
#include <QMutex>
//选择屏幕打印还是输出到文件可以根据这个宏控制或者控制函数调用位置都可以
//#define _DEBUG
//默认调试级别为warning,即小于warning级别的都不会写入日志文件
//只有release版本的时候,才会输出到日志,debug版本正常输出到终端。
namespace QT_LOG
{
    //默认文件名为当前时间命名的log文件
    static int m_LogLevel = 1;
    static QString m_LogFile = QString("%1.log").arg(QDateTime::currentDateTime().toString("yyyyMMddhhmmss"));
    QMutex m_LogMutex;
    void customMessageHandler(QtMsgType type , const QMessageLogContext &context , const QString &msg)
    {
        //设置输出日志级别,小于该级别,将不会写入日志文件,默认是warning级别,即debug信息不会写入日志文件
        if (type < m_LogLevel)
        {
            return;
        }
        QString log_info;
        switch (type)
        {
        case QtDebugMsg:
            log_info = QString("%1[Debug]:").arg(QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss"));
            break;
        case QtWarningMsg:
            log_info = QString("%1[Warning]:").arg(QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss"));
            break;
        case QtCriticalMsg:
            log_info = QString("%1[Critical]:").arg(QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss"));
            break;
        case QtFatalMsg:
            log_info = QString("%1[Fatal]:").arg(QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss"));
            abort();
        case QtInfoMsg:
            log_info = QString("%1[Info]:").arg(QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss"));
            break;
        }
        log_info += QString(context.file) + QString(context.line) + QString("%1").arg(msg);
        //为了线程安全
        m_LogMutex.lock();
        QFile outFile(m_LogFile);
        outFile.open(QIODevice::WriteOnly | QIODevice::Append | QIODevice::Text);
        QTextStream ts(&outFile);
        ts << log_info << endl;
        outFile.close();
        m_LogMutex.unlock();
    }
    //默认调试级别为warning及以上才会写入日志文件,默认log文件名为程序启动时间命名的log文件
    void logInit(QString logFile = "", int logLevel = 1)
    {
        #ifndef _DEBUG  //实现debug版本的时候,输出到终端;release版本的时候输出到日志文件
            if ((logLevel < 0) || (logLevel > 3))
            {
                m_LogLevel = 1;
            }
            else
            {
                m_LogLevel = logLevel;
            }
            if (!logFile.isEmpty())
            {
                m_LogFile = logFile;
            }
            qInstallMessageHandler(customMessageHandler);
        #endif
    }
}
#endif // LOG_H

调用,包含头文件后直接调用参数即可:

QT_LOG::logInit();//或者QT_LOG::logInit("log.txt", 2);QT_LOG::logInit();
//或者
QT_LOG::logInit("log.txt", 2);

3、问题

可能不通Qt版本提供的回调函数有差异,我们通过qInstallMessageHandler进行源码查看,看下回调函数的样式,实现对应的回调函数进行测试即可:

在这里插入图片描述

4、结果与扩展思路

这个是我使用我上面封装的函数后生成的日志文件,每一次启动创建一个,暂时没有考虑定时分割创建新文件等问题,如果有需求可以调整回调函数进行调试:

在这里插入图片描述

针对上面的日志我们还可以加一些定时任务或者直接通过内存数据库通过线程安全的方式直接存储到数据库中,这样便于分布式存储和分析操作日志。

以上就是Qt5实现qDebug日志信息写入日志文件的详细内容,更多关于Qt5 qDebug日志信息写入文件的资料请关注脚本之家其它相关文章!

您可能感兴趣的文章:

相关文章

  • VC中SendMessage和PostMessage的区别

    VC中SendMessage和PostMessage的区别

    这篇文章主要介绍了VC中SendMessage和PostMessage的区别,较为全面的分析了SendMessage和PostMessage运行原理及用法上的不同之处,非常具有实用价值,需要的朋友可以参考下
    2014-10-10
  • C++ 写的UrlEncode和UrlDecode实例

    C++ 写的UrlEncode和UrlDecode实例

    这篇文章主要介绍了C++ 写的UrlEncode和UrlDecode实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • 分析C语言一个简单程序

    分析C语言一个简单程序

    本文主要介绍C语言简单的程序,这里给大家详细介绍C语言代码,对函数概念、头文件、自定义函数等基础信息的讲解,希望能帮助刚刚学习的同学
    2016-07-07
  • C++实现CreatThread函数主线程与工作线程交互的方法

    C++实现CreatThread函数主线程与工作线程交互的方法

    这篇文章主要介绍了C++实现CreatThread函数主线程与工作线程交互的方法,是Windows应用程序设计中非常实用的方法,需要的朋友可以参考下
    2014-10-10
  • C语言简单实现计算字符个数的方法

    C语言简单实现计算字符个数的方法

    这篇文章主要介绍了C语言简单实现计算字符个数的方法,涉及C语言针对字符串的简单遍历与判定技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-07-07
  • 2~62位任意进制转换方法(c++)

    2~62位任意进制转换方法(c++)

    下面小编就为大家带来一篇2~62位任意进制转换方法(c++)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-06-06
  • C++与QML进行数据交互实现方式介绍

    C++与QML进行数据交互实现方式介绍

    迫于无奈开始写android的程序,以前使用QWidget的方式试过,虽然界面可以实现,但是最后调用摄像头时,未能成功,再没有继续。这几天开始使用qml进行尝试,在使用的过程中,其中的一个难点,就是在qml与c++中数据的交互
    2022-09-09
  • C++实现五子棋小游戏

    C++实现五子棋小游戏

    这篇文章主要为大家详细介绍了C++实现五子棋小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-05-05
  • C语言模拟实现atoi函数的实例详解

    C语言模拟实现atoi函数的实例详解

    这篇文章主要介绍了C语言模拟实现atoi函数的实例详解的相关资料,atoi函数,主要功能是将一个字符串转变为整数,这里就实现这样的函数,需要的朋友可以参考下
    2017-08-08
  • C语言实现大数据文件的内存映射机制

    C语言实现大数据文件的内存映射机制

    这篇文章主要介绍了C语言实现大数据文件的内存映射机制的相关资料,需要的朋友可以参考下
    2017-01-01

最新评论