C 语言

关注公众号 jb51net

关闭
首页 > 软件编程 > C 语言 > C++ 轻量级日志系统

从零实现一个 C++ 轻量级日志系统原理与实践指南

作者:Cx330❀

本文从零开始手搓一个简易C++日志工具,详细介绍了日志系统的理念、设计、实现及优化,核心包括:日志格式化时间和等级,基于策略模式实现异步刷新,使用RAII机制实现自动刷新,以及对线程安全和可重入性的深度考量,最后还提供了优化方向,帮助读者完善日志系统

前言:

日志系统可以说是每个程序员都绕不开的话题。在大型C++项目中,日志打印和记录几乎是日常开发中最常用的功能之一 —— 排查Bug、追踪调用链、监控线上服务状态,都离不开一套好用的日志工具。

提到C++日志库,有一些人会想:日志系统到底是怎么实现的? 抛开spdlog这些现成的库,我们自己能不能从零手搓一个可用的日志组件出来?

其实手搓一个简易日志工具并不复杂,核心思路就是 将格式化后的信息输出到不同的目的地,比如控制台、文件等。在这个思路的指引下,我们可以一步步搭建自己的日志系统,并顺便搞懂:

如果你曾对这些问题感到好奇,那这篇文章就是写给你的。下面,我们就从零开始,手搓一个实用且可扩展的C++日志工具。

一. 日志系统的设计理念

1.1 日志的核心组成要素

一条合格的工业级日志,必须包含必选字段可选扩展字段,确保问题可追溯、状态可监控:

本文实现的日志格式如下,完全兼容主流日志库的规范:

[2026-04-16 21:33:18] [DEBUG] [1030871] [Main.cc] [10] - hello world hello Cx330
         日期          日志等级    pid     源文件   行号 -         内容       root   

1.2 日志系统的两大核心阶段

日志的生命周期可拆分为两个完全解耦的阶段,这是我们设计的核心依据:

  1. 日志形成阶段:将时间戳、等级、文件名、行号、用户内容等信息,拼接成一条完整的格式化字符串,与日志输出目的地无关
  2. 日志刷新阶段:将格式化完成的日志字符串,写入到指定目的地(控制台、文件、数据库、网络等),仅关注写入逻辑

两个阶段解耦后,我们可以独立扩展刷新逻辑,而无需修改日志格式化的核心代码,这正是策略模式的最佳应用场景。

1.3 为什么选择策略模式?

二、代码设计:实现一个完整的日志库

日志系统的核心前提是线程安全,同时需要时间戳、日志等级等基础能力支撑,我们先实现这些底层模块。

2.1 RAII 风格互斥锁封装(线程安全基石)

多线程环境下,控制台、日志文件都是临界资源,多个线程同时写入会导致内容交错、乱序,必须通过互斥量保证临界区的原子性。我们基于 Linux 原生的pthread_mutex封装互斥锁,并通过 RAII 机制管理锁的生命周期,避免手动解锁导致的死锁、内存泄漏问题,这也是 C++11 std::lock_guard的核心实现原理。

#ifndef __MUTEX_HPP
#define __MUTEX_HPP
#include <iostream>
#include <pthread.h>
class Mutex
{
public:
    Mutex()
    {
        pthread_mutex_init(&_lock,nullptr);
    }    
    void Lock()
    {
        pthread_mutex_lock(&_lock);
    }
    pthread_mutex_t *Orgin()
    {
        return &_lock;
    }
    void Unlock()
    {
        pthread_mutex_unlock(&_lock);
    }
    ~Mutex()
    {
        pthread_mutex_destroy(&_lock);
    }
private:
    pthread_mutex_t _lock;
};
// 锁的开关
class LockGuard
{
public:
    LockGuard(Mutex *lockp):_lockp(lockp)
    {
        _lockp->Lock();
    }
    ~LockGuard()
    {
        _lockp->Unlock();
    }
private:
    Mutex *_lockp;
};
#endif

核心设计解析

2.2 格式化时间戳模块

时间戳是日志的核心字段,我们需要实现秒级、可重入、格式化的时间戳获取功能。

重点注意:C 标准库的localtime函数是不可重入的,多线程环境下会出现数据错乱,因此必须使用可重入版本localtime_r,它由调用者提供结构体缓冲区,避免了全局静态变量的竞态问题。

// 获取当前时间的字符串表示(格式:YYYY-MM-DD HH:MM:SS)
std::string GetTimeStamp()
{
    // 获取从1970-01-01 UTC到当前时刻的秒数(Unix时间戳)
    time_t timestamp = time(nullptr);
    // 定义tm结构体用于存储分解后的本地时间
    struct tm data_time;
    // 将时间戳转换为本地时间(线程安全版本,localtime_r是POSIX标准)
    localtime_r(&timestamp, &data_time);
    // 缓冲区,用于存放格式化后的时间字符串
    char data_time_str[128];
    // 使用snprintf进行格式化,限制最大写入长度,防止溢出
    // 格式:年-月-日 时:分:秒,各部分不足两位时补零
    snprintf(data_time_str, sizeof(data_time_str), "%4d-%02d-%02d %02d:%02d:%02d",
             // tm_year 从1900年开始计数,需要加1900得到实际年份
             data_time.tm_year + 1900,
             // tm_mon 范围0~11,需要加1转换为实际月份
             data_time.tm_mon + 1,
             data_time.tm_mday,   // 日(1~31)
             data_time.tm_hour,   // 小时(0~23)
             data_time.tm_min,    // 分钟(0~59)
             data_time.tm_sec);   // 秒(0~60,闰秒时可达60)
    // 返回std::string对象,自动拷贝缓冲区内容
    return data_time_str;
}

细节解析

#include <iostream>
#include <memory>
#include <unistd.h>
#include "Logger.hpp"
using namespace LogModule;
// 测试时间戳模块
void testTime()
{
    for(int i = 0; i < 5; i++)
    {
        std::cout << GetTimeStamp() << std::endl;
        sleep(1);
    }
}
int main()
{
    // 1. 测试时间
    testTime();
    return 0;
}

2.3 类型安全的日志等级模块

日志等级用于区分事件的严重程度,我们使用 C++11 的enum class实现类型安全的日志等级,避免普通枚举的隐式类型转换问题,同时提供枚举到字符串的转换能力。

// 日志等级枚举,用于区分事件的严重程度
enum LogLevel
{
    DEBUG,   // 调试信息,仅开发阶段使用,生产环境通常关闭
    INFO,    // 常规信息,如服务启动、正常业务流转
    WARNING, // 警告信息,表示潜在问题,但系统仍可正常运行
    ERROR,   // 运行时错误,某个功能可能受损,需要关注
    FATAL    // 致命错误,程序即将终止(如内存分配失败)
};
// 将日志等级枚举转换为对应的字符串(用于日志输出)
std::string LogLevel2String(LogLevel level)
{
    switch (level)
    {
    case LogLevel::DEBUG:
        return "DEBUG";     // 调试等级
    case LogLevel::INFO:
        return "INFO";      // 信息等级
    case LogLevel::WARNING:
        return "WARNING";   // 警告等级
    case LogLevel::ERROR:
        return "ERROR";     // 错误等级
    case LogLevel::FATAL:
        return "FATAL";     // 致命等级
    default:
        return "UNKNOWN";   // 未匹配到的等级(防护性代码)
    }
}

核心设计解析

#include <iostream>
#include <memory>
#include <unistd.h>
#include "Logger.hpp"
using namespace LogModule;
// 测试日志类枚举类型转字符类型模块
void testEnum()
{
    std::cout << LogLevel2String(LogLevel::DEBUG) << std::endl;
    std::cout << LogLevel2String(LogLevel::INFO) << std::endl;
    std::cout << LogLevel2String(LogLevel::WARNING) << std::endl;
    std::cout << LogLevel2String(LogLevel::ERROR) << std::endl;
    std::cout << LogLevel2String(LogLevel::FATAL) << std::endl;
}
int main()
{
    // 2. 测试枚举类转字符串类型
    testEnum();
    return 0;
}

三. 基于策略模式的日志刷新核心实现

基于策略模式的设计,我们先定义抽象的刷新策略基类,再分别实现控制台和文件两种具体的刷新策略,后续可无限扩展其他策略。

3.1 抽象策略基类 LogStrategy

抽象基类定义了所有刷新策略必须实现的纯虚接口,同时使用虚析构函数确保子类对象能正确析构。

namespace LogModule 
{
    // 3. 刷新策略
    // 基类: 策略模式 (Strategy Pattern)
    // 核心思想:将“日志的产生”与“日志的刷新目的地”解耦。
    // 通过定义统一的接口,使得程序可以在运行时动态决定将日志输出到控制台、文件、数据库或网络。
    class LogStrategy
    {
    public:
        // 虚析构函数:在多态体系中,基类必须拥有虚析构函数。
        // 这样当我们通过基类指针删除派生类对象时,才能确保调用到子类的析构函数,防止内存泄漏。
        virtual ~LogStrategy() = default; // 不在这里析构
        // 核心刷新接口:这是一个纯虚函数。
        // 纯虚函数的核心作用是定义一种“契约”,强制派生类(子类)必须实现具体的逻辑。
        // 不同的子类可以根据自己的策略(如 ConsoleStrategy 或 FileStrategy)来实现不同的刷新行为。
        virtual void SyncLog(const std::string &message) = 0; // 强制子类对其进行重写
    };
}

设计说明

3.2 控制台日志策略 ConsoleLogStrategy

控制台策略负责将日志输出到标准错误流(stderr),核心是保证多线程环境下的输出原子性,避免日志交错。用到了我们自己的互斥锁记得包含对应头文件,我这里就不写了

namespace LogModule
{
    // 策略1: 控制台日志策略
    // 子类:继承自策略基类,用于将日志直接刷新到标准输出(显示器),常用于本地开发与调试 
    class ConsoleLogStrategy: public LogStrategy
    {
    public:
        // 构造函数与析构函数:当前策略不涉及复杂资源申请,故使用默认实现即可 
        ConsoleLogStrategy(){}
        ~ConsoleLogStrategy(){}
        /**
         * @brief 实现具体的日志同步逻辑——刷新到控制台
         * @param message 组装好的完整日志字符串
         */
        void SyncLog(const std::string &message) override // 检查重写的错误
        {
            // 【核心原理】显示器(stdout)在多线程环境下属于“临界资源”。
            // 如果不加保护,多个线程同时调用 std::cout 会导致各条日志的字符在屏幕上发生“交织”或乱码 。
            // 使用自定义的 LockGuard 配合互斥锁,确保这一系列操作的原子性 。
            LockGuard logGuard(&_mutex);
            std::cout << message << std::endl;
        }
    private:
        // 互斥锁:专门用于保护当前控制台输出的原子性,防止并发打印时消息错乱 
        Mutex _mutex;
    };
}

核心细节解析

3.3 文件日志策略 FileLogStrategy

文件策略负责将日志持久化到磁盘文件,核心功能包括:自动创建日志目录、追加模式写入、线程安全保障,使用 C++17 的filesystem库处理目录和文件操作(记得带上对应头文件)。

#include <fstream>
#include <filesystem>
namespace LogModule 
{
    // 定义全局默认路径与文件名常量 
    const static std::string gdefaultlogdir = "./log/";
    const static std::string gdefaultlogfilename = "log.txt";

    // 策略2:文件类日志策略
    // 子类:继承自策略基类,实现将日志持久化到磁盘文件的逻辑 
    class FileLogStrategy: public LogStrategy
    {
    public:
        /**
         * @brief 构造函数:初始化日志路径并确保目录环境就绪
         * @param logdir 日志存储目录
         * @param logfilename 日志文件名称
         */
        FileLogStrategy(const std::string &logdir = gdefaultlogdir, const std::string &logfilename = gdefaultlogfilename)
            :_logdir(logdir),
            _logfilename(logfilename)
        {
            // 【重点】构造阶段即进行加锁保护。
            // 理由:判断目录是否存在并创建目录属于“先检查再执行(Check-Then-Act)”模式,
            // 必须保证这一系列操作的原子性,防止多线程同时创建导致竞态冲突 。
            LockGuard lockGuard(&_mutex);
            
            // 使用 C++17 的 <filesystem> 库进行跨平台路径检查 
            if(std::filesystem::exists(_logdir))
            {
                return;
            }
            else 
            {
                try 
                {
                    // 递归创建目录(类似于 Linux 命令 mkdir -p),如果路径中包含多级不存在的目录会一并创建 
                    std::filesystem::create_directories(_logdir);
                } 
                catch (std::filesystem::filesystem_error &e) 
                {
                    // 捕获文件系统异常(如权限不足、磁盘空间不足等)并输出错误信息 
                    std::cerr << e.what() << std::endl;
                }
            }
        }

        // 析构函数:由于不涉及手动管理的堆内存或特殊文件句柄(使用局部变量流管理),故使用默认实现
        ~FileLogStrategy(){}

        /**
         * @brief 执行具体的日志落盘操作
         * @param message 待写入的完整日志字符串
         */
        void SyncLog(const std::string &message) override
        {
            // 加锁保护:防止多线程同时写入同一文件导致内容交织(Interleaving)乱码 
            LockGuard logGuard(&_mutex);
            
            // 构造完整的目标文件路径
            std::string target = _logdir + _logfilename;
            
            // 以追加模式(std::ios::app)打开文件流:
            // 核心逻辑:保证每条新日志都写在文件末尾,不会覆盖已有日志内容 。
            std::ofstream out(target, std::ios::app); // 追加

            if(!out.is_open()) // 打开文件检查
            {
                return; // 如果因权限或路径问题打开失败,则放弃本次写入,防止程序崩溃
            }
            
            // 将消息流式写入文件,并手动添加换行符以符合日志排版规范 
            out << message << "\n"; // 流式写入
            
            // 文件流离开作用域或显式调用 close 会自动触发刷新并关闭文件
            out.close();
        }

    private:
        std::string _logdir;      // 存储目录路径
        std::string _logfilename; // 存储文件名称
        Mutex _mutex;             // 用于保障当前策略类实例在多线程环境下的线程安全 
    };
}

核心设计解析

测试代码

四. 日志主体类与流式输出设计

完成基础模块和策略模式的实现后,我们来实现日志系统的主体类,核心目标是:兼容 glog 的流式调用风格、自动拼接日志元信息、RAII 自动触发日志刷新

4.1 Logger 主类的整体架构

Logger类是日志系统的对外入口,核心职责包括:

4.2 LogMessage 内部类:RAII 实现日志自动刷新

LogMessageLogger的内部类,是整个日志系统最巧妙的设计:

4.3 完整的 Logger 类实现

#include <memory>
#include <sstream>
#include <unistd.h>
namespace LogModule
{
    // 真正要的日志类
    class Logger
    {
    public: 
        Logger()
        {
            UseConsoleLogStrategy();
        }
        ~Logger(){}
        // 显示器的刷新策略
        void UseConsoleLogStrategy()
        {
            _strategy = std::make_unique<ConsoleLogStrategy>();
        }
        // 文件的刷新策略
        void UseFileLogStrategy()
        {
            _strategy = std::make_unique<FileLogStrategy>();
        }
        // 内部类:一条日志
        // 目标是把一个类对象,变成一个string
        class LogMessage
        {
        public:
            LogMessage(LogLevel level,std::string &filename,int line,Logger &self)
                :_level(level),
                 _curr_time(GetTimeStamp()),
                 _pid(getpid()),
                 _filename(filename),
                 _line(line),
                 _logger(self)
            {   
                std::stringstream ss; 
                ss << "[" << _curr_time << "] "
                   << "[" << LogLevel2String(_level) << "] "
                   << "[" << _pid << "] "
                   << "[" << _filename << "] "
                   << "[" << _line << "] "
                   << "- ";
                _loginfo = ss.str();
            }
            template<typename T>
            LogMessage &operator << (const T &info) 
            {
                std::stringstream ss;
                ss << info;
                _loginfo += ss.str();
                return  *this;
            }
            ~LogMessage() // RAII风格的日志刷新
            {
                if(_logger._strategy)
                {
                    _logger._strategy->SyncLog(_loginfo);
                }
            }
        private:
            LogLevel _level;         // 日志等级
            std::string _curr_time;  // 当前时间
            pid_t _pid;              // 进程pid
            std::string _filename;   // 文件名
            int _line;               // 行号
            std::string _loginfo;    // 一条完整的日志
            Logger &_logger;         // 外部类的引用
        };
        // LogMessage 对象打印日志的时候,故意返回一个临时的 LogMessage对象
        // 为什么要返回临时内部类对象?
        LogMessage operator()(LogLevel level,std::string filename, int line)
        {
            return LogMessage(level,filename,line,*this);
        }
    private:
        std::unique_ptr<LogStrategy> _strategy; // 刷新日志的策略
    };
    Logger logger;
    // 使用宏,包装我们的日志打印过程,宏有一个特点,#define A B,B替换成A
    #define LOG(level) logger(level,__FILE__,__LINE__)
    // 动态调整日志策略
    #define ENABLE_CONSOLE_LOG_STRATEGY() logger.UseConsoleLogStrategy()
    #define ENABLE_FILE_LOG_STRATEGY() logger.UseFileLogStrategy()
}

核心设计深度解析

预定义宏封装:

全局单例:定义全局的logger对象,整个程序共用一个日志实例,避免重复创建,同时保证策略切换全局生效

五. 日志系统的线程安全与可重入性深度解析

六、日志系统源码

6.1 完整Logger.hpp代码

#ifndef __LOGGER_HPP
#define __LOGGER_HPP
#include <iostream>
#include <cstdio>
#include <string>
#include <ctime>
#include <filesystem> // C++17
#include "Mutex.hpp"
#include <fstream>
#include <sstream>
#include <memory>
#include <unistd.h>
namespace LogModule
{
    // 1.获取时间
    std::string GetTimeStamp()
    {
        time_t timestamp = time(nullptr);
        struct tm data_time;
        localtime_r(&timestamp, &data_time);
        char data_time_str[128];
        snprintf(data_time_str, sizeof(data_time_str), "%4d-%02d-%02d %02d:%02d:%02d",
                 data_time.tm_year + 1900, // 从1900开始记的
                 data_time.tm_mon + 1,     // 默认月份从0开始记的
                 data_time.tm_mday,
                 data_time.tm_hour,
                 data_time.tm_min,
                 data_time.tm_sec);
        return data_time_str;
    }
    enum LogLevel
    {
        DEBUG,
        INFO,
        WARNING,
        ERROR,
        FATAL
    };
    // 2.日志等级
    std::string LogLevel2String(LogLevel level)
    {
        switch (level)
        {
        case LogLevel::DEBUG: 
            return "DEBUG";
        case LogLevel::INFO: 
            return "INFO";
        case LogLevel::WARNING: 
            return "WARNING";
        case LogLevel::ERROR: 
            return "ERROR";
        case LogLevel::FATAL: 
            return "FATAL";
        default:
            return "UNKNOWN";
        }
    }
    // 3.日志刷新
    // 基类:策略基类,设置刷新策略的
    class LogStrategy
    {
    public:
        virtual ~LogStrategy() = default;
        virtual void SyncLog(const std::string &logmessage) = 0;
    };
    // 子类:继承纯虚接口类
    // 策略1
    class ConsoleLogStrategy : public LogStrategy
    {
    public:
        ConsoleLogStrategy(){}
        ~ConsoleLogStrategy(){}
        virtual void SyncLog(const std::string &logmessage) override
        {
            LockGuard lockguard(&_mutex); 
            std::cout<<logmessage<<std::endl;
        }
    private:
        Mutex _mutex;
    };
    static const std::string glogdir = "./log/";
    static const std::string glogfilename = "log.log";
    // 子类:继承纯虚接口类
    // 策略2
    class FileLogStrategy : public LogStrategy
    {
    public:
        FileLogStrategy(const std::string &dir = glogdir,const std::string &filename = glogfilename)
            :_logdir(dir),_logfilename(filename)
        {
            // log/log.txt
            LockGuard lockguard(&_mutex);
            if(std::filesystem::exists(_logdir))
            {
                return;
            }
            else
            {
                try
                {
                    std::filesystem::create_directories(_logdir);
                }
                catch (const std::filesystem::filesystem_error &e)
                {
                    std::cerr<< e.what() <<std::endl;
                }
            }
        }
        ~FileLogStrategy()
        {}
        void SyncLog(const std::string &logmessage) override
        {
            std::string target = _logdir + _logfilename;
            std::ofstream out(target,std::ios::app); // 追加写入文件
            if(!out.is_open())
            {
                return;
            }
            // 方法1:
            // out.write(logmessage.c_str(), logmessage.size());
            // out.write("\n", 1); // 写入换行符
            // 方法2:
            // std::string line = logmessage + '\n';
            // out.write(line.c_str(), line.size());
            // 方法3:
            out << logmessage << '\n';
            out.close();
        }
    private:
        std::string _logdir;
        std::string _logfilename; // ./log/XXX.log
        Mutex _mutex;
    };
    // 真正要的日志类
    class Logger
    {
    public: 
        Logger()
        {
            UseConsoleLogStrategy();
        }
        ~Logger(){}
        // 显示器的刷新策略
        void UseConsoleLogStrategy()
        {
            _strategy = std::make_unique<ConsoleLogStrategy>();
        }
        // 文件的刷新策略
        void UseFileLogStrategy()
        {
            _strategy = std::make_unique<FileLogStrategy>();
        }
        // 内部类:一条日志
        // 目标是把一个类对象,变成一个string
        class LogMessage
        {
        public:
            LogMessage(LogLevel level,std::string &filename,int line,Logger &self)
                :_level(level),
                 _curr_time(GetTimeStamp()),
                 _pid(getpid()),
                 _filename(filename),
                 _line(line),
                 _logger(self)
            {   
                std::stringstream ss; 
                ss << "[" << _curr_time << "] "
                   << "[" << LogLevel2String(_level) << "] "
                   << "[" << _pid << "] "
                   << "[" << _filename << "] "
                   << "[" << _line << "] "
                   << "- ";
                _loginfo = ss.str();
            }
            template<typename T>
            LogMessage &operator << (const T &info) 
            {
                std::stringstream ss;
                ss << info;
                _loginfo += ss.str();
                return  *this;
            }
            ~LogMessage() // RAII风格的日志刷新
            {
                if(_logger._strategy)
                {
                    _logger._strategy->SyncLog(_loginfo);
                }
            }
        private:
            LogLevel _level;         // 日志等级
            std::string _curr_time;  // 当前时间
            pid_t _pid;              // 进程pid
            std::string _filename;   // 文件名
            int _line;               // 行号
            std::string _loginfo;    // 一条完整的日志
            Logger &_logger;         // 外部类的引用
        };
        // LogMessage 对象打印日志的时候,故意返回一个临时的 LogMessage对象
        // 为什么要返回临时内部类对象?
        LogMessage operator()(LogLevel level,std::string filename, int line)
        {
            return LogMessage(level,filename,line,*this);
        }
    private:
        std::unique_ptr<LogStrategy> _strategy; // 刷新日志的策略
    };
    Logger logger;
    // 使用宏,包装我们的日志打印过程,宏有一个特点,#define A B,B替换成A
    #define LOG(level) logger(level,__FILE__,__LINE__)
    // 动态调整日志策略
    #define ENABLE_CONSOLE_LOG_STRATEGY() logger.UseConsoleLogStrategy()
    #define ENABLE_FILE_LOG_STRATEGY() logger.UseFileLogStrategy()
}
#endif

6.2 完整测试代码

#include <iostream>
#include <unistd.h>
#include <pthread.h>
#include "Logger.hpp" // 我们实现的日志头文件
using namespace LogModule;
// 多线程测试函数:10个线程同时打印日志
void *thread_log_test(void *arg)
{
    char *thread_name = (char *)arg;
    for (int i = 0; i < 5; i++)
    {
        LOG(LogLevel::INFO) << thread_name << " 执行日志打印, 循环次数: " << i;
        usleep(1000);
    }
    return nullptr;
}
int main()
{
    // 1. 基础控制台日志输出
    std::cout << "===== 控制台日志测试 =====" << std::endl;
    ENABLE_CONSOLE_LOG_STRATEGY();
    LOG(LogLevel::DEBUG) << "这是DEBUG调试日志, 数值: " << 3.14159;
    LOG(LogLevel::INFO) << "这是INFO常规日志, 服务启动成功";
    LOG(LogLevel::WARNING) << "这是WARNING警告日志, 配置缺失, 使用默认值";
    LOG(LogLevel::ERROR) << "这是ERROR错误日志, 文件读取失败";
    LOG(LogLevel::FATAL) << "这是FATAL致命日志, 内存耗尽, 服务退出";
    // 2. 切换为文件日志策略
    std::cout << "\n===== 文件日志测试 =====" << std::endl;
    ENABLE_FILE_LOG_STRATEGY();
    LOG(LogLevel::INFO) << "切换为文件日志策略, 日志将持久化到./log/log.txt";
    LOG(LogLevel::DEBUG) << "文件日志测试, 支持链式拼接: " << "字符串 " << 1234 << " 浮点数 " << 2.71828;
    // 3. 多线程线程安全测试
    std::cout << "\n===== 多线程日志测试 =====" << std::endl;
    pthread_t t1, t2, t3, t4;
    pthread_create(&t1, nullptr, thread_log_test, (void *)"thread-1");
    pthread_create(&t2, nullptr, thread_log_test, (void *)"thread-2");
    pthread_create(&t3, nullptr, thread_log_test, (void *)"thread-3");
    pthread_create(&t4, nullptr, thread_log_test, (void *)"thread-4");
    // 等待所有线程执行完毕
    pthread_join(t1, nullptr);
    pthread_join(t2, nullptr);
    pthread_join(t3, nullptr);
    pthread_join(t4, nullptr);
    LOG(LogLevel::INFO) << "多线程日志测试完成, 无乱序、无交错";
    return 0;
}

6.3 优化方向

写在最后

至此,我们完整剖析了一个现代 C++ 流式日志系统的核心设计:从日志等级的枚举定义,到策略模式解耦输出目标,再到 RAII + 临时对象实现的自动刷新,以及线程安全与可重入性的深度考量。

这个日志库虽然只有短短几百行代码,却凝聚了策略模式、RAII、智能指针、临时对象生命周期、流式接口、宏与预定义标识符等 C++ 关键思想。更重要的是,通过分析它的线程安全漏洞和性能瓶颈,我们更能理解并发编程的复杂性以及工程落地必须权衡的取舍

当然,任何代码都不是完美的。你已看到它的十项优化方向——从修复数据竞争到支持异步日志,每一点改进都能让这个轮子更滚得更远。如果你正在为自己项目的日志组件苦恼,不妨从这份代码出发,增加等级过滤、持久的文件流、可配置格式等特性,打造一个真正生产就绪的日志库。

最后,感谢你跟随本文深入到这个看似简单却暗藏玄机的模块中。日志是系统的“黑匣子”,好的日志设计能让你在排查问题时事半功倍。希望这份解析能为你带来实实在在的启发,也欢迎在评论区分享你的日志系统实践或疑问。

到此这篇关于从零实现一个 C++ 轻量级日志系统原理与实践指南的文章就介绍到这了,更多相关 C++ 轻量级日志系统内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文