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

相关文章

使用Sentinel自定义返回和实现区分来源方式

《使用Sentinel自定义返回和实现区分来源方式》:本文主要介绍使用Sentinel自定义返回和实现区分来源方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Sentinel自定义返回和实现区分来源1. 自定义错误返回2. 实现区分来源总结Sentinel自定

SpringBoot日志配置SLF4J和Logback的方法实现

《SpringBoot日志配置SLF4J和Logback的方法实现》日志记录是不可或缺的一部分,本文主要介绍了SpringBoot日志配置SLF4J和Logback的方法实现,文中通过示例代码介绍的非... 目录一、前言二、案例一:初识日志三、案例二:使用Lombok输出日志四、案例三:配置Logback一

golang 日志log与logrus示例详解

《golang日志log与logrus示例详解》log是Go语言标准库中一个简单的日志库,本文给大家介绍golang日志log与logrus示例详解,感兴趣的朋友一起看看吧... 目录一、Go 标准库 log 详解1. 功能特点2. 常用函数3. 示例代码4. 优势和局限二、第三方库 logrus 详解1.

C++ 中的 if-constexpr语法和作用

《C++中的if-constexpr语法和作用》if-constexpr语法是C++17引入的新语法特性,也被称为常量if表达式或静态if(staticif),:本文主要介绍C++中的if-c... 目录1 if-constexpr 语法1.1 基本语法1.2 扩展说明1.2.1 条件表达式1.2.2 fa

如何自定义Nginx JSON日志格式配置

《如何自定义NginxJSON日志格式配置》Nginx作为最流行的Web服务器之一,其灵活的日志配置能力允许我们根据需求定制日志格式,本文将详细介绍如何配置Nginx以JSON格式记录访问日志,这种... 目录前言为什么选择jsON格式日志?配置步骤详解1. 安装Nginx服务2. 自定义JSON日志格式各

Android自定义Scrollbar的两种实现方式

《Android自定义Scrollbar的两种实现方式》本文介绍两种实现自定义滚动条的方法,分别通过ItemDecoration方案和独立View方案实现滚动条定制化,文章通过代码示例讲解的非常详细,... 目录方案一:ItemDecoration实现(推荐用于RecyclerView)实现原理完整代码实现

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++从序列容器中删除元素的四种方法

《C++从序列容器中删除元素的四种方法》删除元素的方法在序列容器和关联容器之间是非常不同的,在序列容器中,vector和string是最常用的,但这里也会介绍deque和list以供全面了解,尽管在一... 目录一、简介二、移除给定位置的元素三、移除与某个值相等的元素3.1、序列容器vector、deque

C++常见容器获取头元素的方法大全

《C++常见容器获取头元素的方法大全》在C++编程中,容器是存储和管理数据集合的重要工具,不同的容器提供了不同的接口来访问和操作其中的元素,获取容器的头元素(即第一个元素)是常见的操作之一,本文将详细... 目录一、std::vector二、std::list三、std::deque四、std::forwa

C++字符串提取和分割的多种方法

《C++字符串提取和分割的多种方法》在C++编程中,字符串处理是一个常见的任务,尤其是在需要从字符串中提取特定数据时,本文将详细探讨如何使用C++标准库中的工具来提取和分割字符串,并分析不同方法的适用... 目录1. 字符串提取的基本方法1.1 使用 std::istringstream 和 >> 操作符示