Qt浅谈之二十一log调试日志
一、简单介绍
近期因调试code时,想了解程序的流程,但苦于没有一个简易的日志记录,不停使用qDebug打印输出,而终于提交代码时得去多次删除信息打印,有时还会出现新改动的代码分不清是哪些部分。而使用#ifdef _DEBUG又比較烦这套,因此写了些简单的日志,方便排除问题,临时不能用于多线程中,以后须要再补充。
二、具体解释
1、追踪函数
#ifdef _DEBUG_PRINT
#define DEBUGPRINT DEBUGInfo printinfo(__FILE__, __LINE__, __FUNCTION__);
#else
#define DEBUGPRINT
#endif class DEBUGInfo
{
public:
DEBUGInfo(QString file, int line, QString func);
~DEBUGInfo(); private:
QString fileName;
int fileLine;
QString funcName;
};
DEBUGInfo::DEBUGInfo(QString file, int line, QString func)
:fileName(file)
,fileLine(line)
,funcName(func)
{
QString result = "";
QString beginTime = QDateTime::currentDateTime().toString("[yyyy-dd-MM hh:mm:ss.zzz] ");
result += beginTime + "Enter:{" + fileName + ":" + QString::number(fileLine) + "---" + funcName + "}";
qDebug() << result.toStdString().c_str();
}
DEBUGInfo::~DEBUGInfo()
{
QString result = "";
QString beginTime = QDateTime::currentDateTime().toString("[yyyy-dd-MM hh:mm:ss.zzz] ");
result += beginTime + "Leave:{" + fileName + ":" + QString::number(fileLine) + "---" + funcName + "}";
qDebug() << result.toStdString().c_str();
}
使用:在须要查看的函数中增加DEBUGPRINT就可以。输出文件名称,所在的行和函数名字。
void HelloWorld()
{
DEBUGPRINT
}
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdGFpeWFuZzE5ODc5MTI=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">
2、信息打印和写日志文件
enum LOGLEVEL{
LOG_DEBUG = 0, /**< Debug >**/
LOG_INFO, /**< Info >**/
LOG_WARN, /**< Warn >**/
LOG_ERROR /**< Error >**/
};
class LogWriter{
public:
static LogWriter* getLogCenter();
void PrintLog(LOGLEVEL level, const char* msg, ...);
void SaveFileLog(LOGLEVEL level, const char* msg, ...);
void setLogPath(QString logPath); //defalut: current path
void setLogLevel(LOGLEVEL logLevel); //defalut: LOG_DEBUG
private:
static LogWriter * _logCenter;
explicit LogWriter();
~LogWriter();
private:
QString _logPath;
LOGLEVEL _logLevel;
private:
QString getLevelStr(LOGLEVEL level);
};
LogWriter *LogWriter::getLogCenter()
{
if (_logCenter == NULL)
_logCenter = new LogWriter;
return _logCenter;
} LogWriter::LogWriter()
{
_logLevel = LOG_DEBUG;
_logPath = QDir::currentPath();
} LogWriter::~LogWriter()
{
if (_logCenter) {
delete _logCenter;
_logCenter = NULL;
}
} void LogWriter::PrintLog(LOGLEVEL level, const char *msg, ...)
{
if (level < _logLevel) return; //low level
char logBuffer[8192] = {0};
va_list vl_fmt; //buffer
va_start(vl_fmt, msg);
vsprintf(logBuffer, msg, vl_fmt);
va_end(vl_fmt); QString fileTime = "";
QString logTime = "";
QString logLevel = getLevelStr(level);
fileTime = QDateTime::currentDateTime().toString("yyyyddMM");
logTime = QDateTime::currentDateTime().toString("yyyy-dd-MM hh:mm:ss.zzz");
qDebug("[%s] [%s] {%s}", logTime.toStdString().c_str(), logLevel.toStdString().c_str(), logBuffer);
} void LogWriter::SaveFileLog(LOGLEVEL level, const char *msg, ...)
{
if (level < _logLevel) return; //low level
char logBuffer[8192] = {0};
va_list vl_fmt; //buff
va_start(vl_fmt, msg);
vsprintf(logBuffer, msg, vl_fmt);
va_end(vl_fmt); QString logTime = "";
QString fileTime = "";
fileTime = QDateTime::currentDateTime().toString("yyyyddMM");
logTime = QDateTime::currentDateTime().toString("[yyyy-dd-MM hh:mm:ss.zzz]");
QString logLevel = getLevelStr(level);
QString logFile = _logPath;
if (logFile.right(1) != "/") {
logFile += "/";
}
QDir mDir(logFile);
if (!mDir.exists()) {
mDir.mkpath(logFile);
}
logFile += "isoft_";
logFile += fileTime;
logFile += ".log"; QFile file(logFile);
file.open(QIODevice::ReadWrite | QIODevice::Append | QIODevice::Text);
QTextStream out(&file);
out << logTime << " [" << logLevel << "] " << "{" << logBuffer << "}" << endl;
file.close();
} QString LogWriter::getLevelStr(LOGLEVEL level)
{
switch(level) {
case LOG_DEBUG: return "LOG_DEBUG"; break;
case LOG_INFO: return "LOG_INFO"; break;
case LOG_WARN: return "LOG_WARN"; break;
case LOG_ERROR: return "LOG_ERROR"; break;
}
} void LogWriter::setLogPath(QString logPath)
{
_logPath = logPath;
} void LogWriter::setLogLevel(LOGLEVEL logLevel)
{
_logLevel = logLevel;
}
能够设置日志的级别和日志文件的路径。
在函数中使用:
LogWriter::getLogCenter()->PrintLog(LOG_DEBUG, "hello world");
LogWriter::getLogCenter()->PrintLog(LOG_INFO, "%s:%s,%d", "hello", "world", 1234);
LogWriter::getLogCenter()->SaveFileLog(LOG_WARN, "hello world");
LogWriter::getLogCenter()->SaveFileLog(LOG_ERROR, "%s:%s,%d", "hello", "world", 1234);
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdGFpeWFuZzE5ODc5MTI=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">
也会在当前路径下生成文件。
3、Qt自带样例
Qt官方样例:
#include <qapplication.h>
#include <stdio.h>
#include <stdlib.h> void myMessageOutput(QtMsgType type, const char *msg)
{
switch (type) {
case QtDebugMsg:
fprintf(stderr, "Debug: %s\n", msg);
break;
case QtWarningMsg:
fprintf(stderr, "Warning: %s\n", msg);
break;
case QtCriticalMsg:
fprintf(stderr, "Critical: %s\n", msg);
break;
case QtFatalMsg:
fprintf(stderr, "Fatal: %s\n", msg);
abort();
}
} int main(int argc, char **argv)
{
qInstallMsgHandler(myMessageOutput);
QApplication app(argc, argv);
...
return app.exec();
}
自己定义改动:
void outputMessage(QtMsgType type, const char *msg)
{
static QMutex mutex;
mutex.lock(); QString text;
switch(type)
{
case QtDebugMsg:
text = QString("Debug:");
break; case QtWarningMsg:
text = QString("Warning:");
break; case QtCriticalMsg:
text = QString("Critical:");
break; case QtFatalMsg:
text = QString("Fatal:");
abort();
}
QString message = QString("[%1] %2 %3").arg(QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss ddd")).arg(text).arg(msg); QFile file("log.txt");
file.open(QIODevice::WriteOnly | QIODevice::Append);
QTextStream text_stream(&file);
text_stream << message << endl;
file.flush();
file.close(); mutex.unlock();
}
qInstallMsgHandler(outputMessage);
qWarning("This is a warning message");
qCritical("This is a critical message");
//qInstallMsgHandler(0) //To restore the message handler, call
执行结果:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">
三、总结
(1)本文仅仅是一个简单的日志记录。还能够设计成异步的多线程式的,甚至能够增加到线程池,对性能要求较高的系统还得考虑文件的大小控制、存储空间的控制、文件的级别控制、文件的日期控制和自己主动清除等操作。
(2)本人思路有限,若有更好的设计建议。也可发邮件沟通,在此先感谢。邮箱地址yang.ao@i-soft.com.cn。
Qt浅谈之二十一log调试日志的更多相关文章
- Qt浅谈之二十七进程间通信之QtDBus
一.简介 DBus的出现,使得Linux进程间通信更加便捷,不仅可以和用户空间应用程序进行通信,而且还可以和内核的程序进行通信,DBus使得Linux变得更加智能,更加具有交互性. DB ...
- Qt浅谈之二十App自动重启及关闭子窗口
一.简介 最近因项目需求,Qt程序一旦检测到错误,要重新启动,自己是每次关闭主窗口的所有子窗口但有些模态框会出现问题,因此从网上总结了一些知识点,以备以后的应用. 二.详解 1.Qt结构 int ma ...
- Qt浅谈之二十App自动重启及关闭子窗口(六种方法)
一.简介 最近因项目需求,Qt程序一旦检测到错误,要重新启动,自己是每次关闭主窗口的所有子窗口但有些模态框会出现问题,因此从网上总结了一些知识点,以备以后的应用. 二.详解 1.Qt结构 int ma ...
- Qt浅谈之二十六图片滑动效果
一.简介 博客中发现有作者写的仿360的代码,觉得其中图片滑动的效果很有意思,特提取其中的代码.并加上类似mac的画面移动的动画效果. 二.详解 1.代码一:界面滑动(QWidget) (1)slid ...
- Qt浅谈之二:钟表(时分秒针)
一.简介 QT编写的模拟时钟,demo里的时钟只有时针和分针,在其基础上添加了秒针,构成了一个完整的时钟.能对2D绘图中坐标系统.平移变换(translate).比例变换(scale).旋转变换(ro ...
- Qt浅谈之二十七进程间通信之QtDBus good
一.简介 DBus的出现,使得Linux进程间通信更加便捷,不仅可以和用户空间应用程序进行通信,而且还可以和内核的程序进行通信,DBus使得Linux变得更加智能,更加具有交互性. DB ...
- Qt浅谈内存泄露(总结)
Qt浅谈内存泄露(总结) 来源 http://blog.csdn.net/taiyang1987912/article/details/29271549 一.简介 Qt内存管理机制:Qt 在内部能够维 ...
- Qt浅谈之总结(整理)
Qt浅谈之总结(整理) 来源 http://blog.csdn.net/taiyang1987912/article/details/32713781 一.简介 QT的一些知识点总结,方便以后查阅. ...
- Android开发-浅谈架构(二)
写在前面的话 我记得有一期罗胖的<罗辑思维>中他提到 我们在这个碎片化 充满焦虑的时代该怎么学习--用30%的时间 了解70%该领域的知识然后迅速转移芳草鲜美的地方 像游牧民族那样.原话应 ...
随机推荐
- java 协程框架kilim
http://phl.iteye.com/blog/2247112 http://chen-tao.github.io/2015/10/02/kilim-work-way/ 待丰富
- 小白用shiro(2)
本文来自网易云社区 作者:王飞 以上的配置走完以后就可以用,下面讲讲个人需求,以及踩过的坑: 1.如何修改cookie的名称,默认名称"rememberMe"太丑了有木有? 首先丢 ...
- appium+python自动化-微信公众号webview操作
前言 上一篇已经解决切换到微信公众号的webview上了,但是定位webview上元素的时候一直提示找不到,打印page_source也找不到页面上的元素,这个问题困扰了一整天,还好最后找到了原因, ...
- Android开发调试无法连接到夜神模拟器的解决方法
Android开发调试无法连接到夜神模拟器的解决方法: 一般原因是adb的版本不一致造成的!!!!!换成一样的就可以了. 在网上看到的方法,特记录下来: 1.任务管理器里看下,adb.exe以及nox ...
- 【转】OPC远程访问相关配置信息
原文:http://blog.gkong.com/kking_25653.ashx 对于远程访问OPC服务器,需要在客户和服务器计算机上都进行DCOM设置,本文提供一些具体配置方法.(by Kevin ...
- javascript前端下载
<html> <head> <title>测试标题</title> </head> <body> <div> 测试页 ...
- XML文件中<return_code><![CDATA[SUCCESS]]></return_code>中CDATA的用法
转义字符不合法的XML字符必须被替换为相应的实体. 如果在XML文档中使用类似"<" 的字符, 那么解析器将会出现错误,因为解析器会认为这是一个新元素的开始.所以不应该象下面 ...
- BZOJ 3757 苹果树 ——莫队算法
挺好的一道题目,怎么就没有版权了呢?大数据拍过了,精神AC.... 发现几种颜色这性质比较垃圾,不可加,莫队硬上. %了一发popoqqq大神的博客, 看了一波VFK关于糖果公园的博客, 又找了wjm ...
- [luoguP2157] [SDOI2009]学校食堂Dining(状压DP)
传送门 这种鬼畜的状压DP...第一次见 看到 0 <= Bi <= 7 就应该想到状态压缩,然而此题实在太鬼畜,想到也没什么乱用 f[i][j][k]表示前i-1个人全部吃完,i~i+7 ...
- BZOJ 2463: [中山市选2009]谁能赢呢?【博弈】
这题不科学~~本以为鬼谷子的钱袋是能在BZOJ写的最短的程序了,这题还要短…..好吧,思考难度神马的还是有点的(至少对我这种蒟蒻来说).很明显这是道博弈论的题目,在纸上画出了n=1~4的博弈树,发现b ...