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

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

C#中通过Response.Headers设置自定义参数的代码示例

《C#中通过Response.Headers设置自定义参数的代码示例》:本文主要介绍C#中通过Response.Headers设置自定义响应头的方法,涵盖基础添加、安全校验、生产实践及调试技巧,强... 目录一、基础设置方法1. 直接添加自定义头2. 批量设置模式二、高级配置技巧1. 安全校验机制2. 类型

C++读写word文档(.docx)DuckX库的使用详解

《C++读写word文档(.docx)DuckX库的使用详解》DuckX是C++库,用于创建/编辑.docx文件,支持读取文档、添加段落/片段、编辑表格,解决中文乱码需更改编码方案,进阶功能含文本替换... 目录一、基本用法1. 读取文档3. 添加段落4. 添加片段3. 编辑表格二、进阶用法1. 文本替换2

C++中处理文本数据char与string的终极对比指南

《C++中处理文本数据char与string的终极对比指南》在C++编程中char和string是两种用于处理字符数据的类型,但它们在使用方式和功能上有显著的不同,:本文主要介绍C++中处理文本数... 目录1. 基本定义与本质2. 内存管理3. 操作与功能4. 性能特点5. 使用场景6. 相互转换核心区别

Java 日志中 Marker 的使用示例详解

《Java日志中Marker的使用示例详解》Marker是SLF4J(以及Logback、Log4j2)提供的一个接口,它本质上是一个命名对象,你可以把它想象成一个可以附加到日志语句上的标签或戳... 目录什么是Marker?为什么使用Markejavascriptr?1. 精细化的过滤2. 触发特定操作3

linux查找java项目日志查找报错信息方式

《linux查找java项目日志查找报错信息方式》日志查找定位步骤:进入项目,用tail-f实时跟踪日志,tail-n1000查看末尾1000行,grep搜索关键词或时间,vim内精准查找并高亮定位,... 目录日志查找定位在当前文件里找到报错消息总结日志查找定位1.cd 进入项目2.正常日志 和错误日

SpringBoot AspectJ切面配合自定义注解实现权限校验的示例详解

《SpringBootAspectJ切面配合自定义注解实现权限校验的示例详解》本文章介绍了如何通过创建自定义的权限校验注解,配合AspectJ切面拦截注解实现权限校验,本文结合实例代码给大家介绍的非... 目录1. 创建权限校验注解2. 创建ASPectJ切面拦截注解校验权限3. 用法示例A. 参考文章本文

C++右移运算符的一个小坑及解决

《C++右移运算符的一个小坑及解决》文章指出右移运算符处理负数时左侧补1导致死循环,与除法行为不同,强调需注意补码机制以正确统计二进制1的个数... 目录我遇到了这么一个www.chinasem.cn函数由此可以看到也很好理解总结我遇到了这么一个函数template<typename T>unsigned

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法