c++ 自定义Logger 日志类

2024-02-13 02:20
文章标签 c++ 日志 自定义 logger

本文主要是介绍c++ 自定义Logger 日志类,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Logger 日志类

线程安全的日志组件

默认保存到文件,并支持回调函数,比如显示到界面

#ifndef LOGGER_H
#define LOGGER_H#include <iostream>
#include <sstream>
#include <mutex>
#include <thread>
#include <iomanip>
#include <fstream>
#include <string>
#include <condition_variable>
#include <thread>
#include <queue>
#include <chrono>
#include <ctime>
#include <functional>// 线程安全的日志组件
class  Logger
{
public:enum class LogLevel{Unknown=0,  //未知信息Debug,      //调试信息Info,       //普通信息Trace,      //详细信息Warning,    //警告信息Error,      //错误信息,但程序可以继续运行。Fatal,      //出现无法恢复的错误,可能导致程序崩溃。Panic      //出现严重错误,必须立即停止程序运行。};Logger();Logger(const std::string& path);~Logger();void set_file_name(const std::string& path);//添加回调函数void set_callback(const std::function<void(std::pair<LogLevel, std::string>)>& func);//输出日志void add_log(LogLevel level, const std::string& message);//转字符串static std::string level2string(LogLevel level);
private://工作线程void worker();//获取当前时间std::string get_current_timestamp();//结束线程void stop();
private:std::mutex _mutex;std::condition_variable _condition;std::queue<std::pair<LogLevel, std::string>> _queue;std::thread _worker;std::string _path;std::ofstream _file;bool _stop;std::function<void(std::pair<LogLevel, std::string>)> _unction;
};#endif // LOGGER_H
 #include "logger.h"Logger::Logger(){}
Logger::Logger(const std::string& path) : _path(path),_stop(false), _worker(&Logger::worker, this)
{_file.open(_path, std::ios_base::app);
}void Logger::set_file_name(const std::string& path)
{_path = path;_file.open(_path, std::ios_base::app);
}
Logger::~Logger()
{stop();_worker.join();//关闭文件if (_file.is_open()){_file.close();}
}
//添加回调函数 输出
void Logger::set_callback(const std::function<void(std::pair<LogLevel, std::string>)>& func)
{_unction = func;
}
std::string Logger::level2string(LogLevel level)
{switch (level){case LogLevel::Debug: return "Debug";case LogLevel::Info: return "Info";case LogLevel::Trace: return "Trace";case LogLevel::Warning: return "Warning";case LogLevel::Error: return "Error";case LogLevel::Fatal: return "Fatal";case LogLevel::Panic: return "Panic";default: return "Unknown";}
}
//输出日志
void Logger::add_log(LogLevel level, const std::string& message)
{std::lock_guard<std::mutex> lock(_mutex);_queue.emplace(level, message);_condition.notify_one();
}//工作线程
void Logger::worker()
{for(;;){std::pair<LogLevel, std::string> pair;{std::unique_lock<std::mutex> lock(_mutex);_condition.wait(lock, [this] { return _stop || !_queue.empty(); });if (_stop && _queue.empty()){break;}pair = _queue.front();_queue.pop();}// //输出到文件// std::ofstream out(_path, std::ios_base::app);// if (!out.good())// {//     continue;// }// out << get_current_timestamp() <<"  "<<level2string(pair.first) <<":"<< pair.second << std::endl;// out.close();// 如果文件流不可用,则重新尝试打开if (!_file.is_open()){_file.open(_path, std::ios_base::app);if (!_file.is_open()){// 打开文件失败,继续下一个循环continue;}}// 输出日志消息到文件流_file << get_current_timestamp() << "  " << level2string(pair.first) << ":" << pair.second << std::endl;_file.flush(); // 刷新缓冲区if(_unction){_unction(pair);}}
}
//获取当前时间
std::string Logger::get_current_timestamp()
{auto now = std::chrono::system_clock::now();auto time = std::chrono::system_clock::to_time_t(now);auto ms = std::chrono::duration_cast<std::chrono::milliseconds>(now.time_since_epoch()) % 1000;std::stringstream ss;ss << std::put_time(std::localtime(&time), "%Y-%m-%d %H:%M:%S") << '.' << std::setfill('0') << std::setw(3) << ms.count();return ss.str();
}//结束线程
void Logger::stop()
{std::lock_guard<std::mutex> lock(_mutex);_stop = true;_condition.notify_one();
}

比如:
使用qt界面,接收日志,,通过接收 logReceived 信号,获取日志信息显示。这里QtConcurrent::run 可以方便接收信息,触发异步信号,不阻塞界面。不然界面会卡死,更新UI只能在一个线程中


// 增加日志回调
void StreamManagerWidget::log_function(std::pair<Logger::LogLevel, std::string> pair)
{// 在后台线程执行日志记录任务QtConcurrent::run(&_threadPool,[this, pair](){// 将日志信息格式化为 QStringQString logMessage = QString::fromStdString(Logger::level2string(pair.first) +": "+ pair.second);// 发送信号,在主线程中更新 UIemit logReceived(logMessage);});
}

这篇关于c++ 自定义Logger 日志类的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

SpringBoot+EasyExcel实现自定义复杂样式导入导出

《SpringBoot+EasyExcel实现自定义复杂样式导入导出》这篇文章主要为大家详细介绍了SpringBoot如何结果EasyExcel实现自定义复杂样式导入导出功能,文中的示例代码讲解详细,... 目录安装处理自定义导出复杂场景1、列不固定,动态列2、动态下拉3、自定义锁定行/列,添加密码4、合并

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

C++中NULL与nullptr的区别小结

《C++中NULL与nullptr的区别小结》本文介绍了C++编程中NULL与nullptr的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编... 目录C++98空值——NULLC++11空值——nullptr区别对比示例 C++98空值——NUL

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window

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

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

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

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

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方