Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >C++实现-带有颜色输出的简单日志类

C++实现-带有颜色输出的简单日志类

作者头像
lexingsen
发布于 2022-02-25 01:08:38
发布于 2022-02-25 01:08:38
76901
代码可运行
举报
文章被收录于专栏:乐行僧的博客乐行僧的博客
运行总次数:1
代码可运行
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#ifndef _LIGHT_LOG_H
#define _LIGHT_LOG_H

#include <iostream>
#include <ostream>
#include <fstream>
#include <sstream>
#include <string>
#include <memory>
#include <mutex>
#include <cstdlib>

namespace llog {

enum LOG_LEVEL {
    LOG_LEVEL_INFO,
    LOG_LEVEL_WARNING,
    LOG_LEVEL_ERROR
};


// 定义不同类型对应的颜色类型
#define COLOR_ERRO 31 //31
#define COLOR_INFO 37 //37
#define COLOR_WARN 33 //33
#define COLOR_FILE 35 //35
#define COLOR_LINE 32 //32
#define COLOR_FUNC 36 //36
#define DELIMITER " "

class Message {
private:
    std::string head; // 消息头
    std::string head_color; // 带颜色的消息头
    std::string msg; // 消息
    std::string colorStr(int color, int bold, const std::string & delimiter, const std::string & t, bool if_color = true) {
        std::stringstream ss;
        if (if_color)
            ss << "\033[" << bold << ";" << color << "m";
        ss << t;
        if (if_color)
            ss << "\033[0m";
        ss << delimiter;
		return ss.str();
    }
    // 默认开启颜色输出
    std::string logHeader(int level, std::string file, std::string function, int linenumber, bool if_color = true) {
        std::stringstream ss;
        std::stringstream st;
        int color = 39;
        if (level==LOG_LEVEL_ERROR) { color = COLOR_ERRO; st << "[ERRO]"; }
		if (level==LOG_LEVEL_INFO) { color = COLOR_INFO; st << "[INFO]"; }
		if (level==LOG_LEVEL_WARNING) { color = COLOR_WARN; st << "[WARN]"; }
        ss << colorStr(color, 1, DELIMITER, st.str(), if_color);
		
		
		
		// 格式化时间戳
        time_t rawtime;
		struct tm* timeinfo;
		char timeBuffer[30];
		time(&rawtime);
		timeinfo = localtime(&rawtime);
		strftime(timeBuffer, 30, "%Y/%m/%d-%H:%M:%S", timeinfo); 
		ss << colorStr(color, 0, DELIMITER, timeBuffer, if_color);

        int pPos = 0;
		for (int i = file.length()-1; i > 0; i--)
			if (file.substr(i,1)==std::string("/"))
			{ pPos = i+1; break; }
		file = file.substr(pPos,file.length()-pPos);
		ss << colorStr(COLOR_FILE, 0, ":", file, if_color); //35
		ss << colorStr(COLOR_LINE, 1, ":", std::to_string(linenumber), if_color); //32
		ss << colorStr(COLOR_FUNC, 0, DELIMITER, function, if_color); //36

        return ss.str();
    }
public:
    Message(const int level = 0, std::string file = nullptr, std::string function = nullptr, const int linenumber = 0) {
        head_color = logHeader(level, file, function, linenumber);
        head = logHeader(level, file, function, linenumber, false);
    }
    
    template <typename T> 
	Message& operator<<(const T & t) {
        std::stringstream ss;
        ss << t;
        msg += ss.str();
        return *this;
    }
    std::string get_string(bool if_color = true) const {
        if (if_color)
            return head_color + msg;
        return head + msg;
    }
};

// Singleton Class
class Logger {
private:
    static std::unique_ptr<Logger>          m_instance_;
    LOG_LEVEL                               m_level_;
    static std::mutex                       mtx_;
    bool                                    console_;
    bool                                    file_;
    std::ofstream                           log_file;
    Logger() {
        if (getenv("LLOG_LV") == NULL)
            m_level_ = LOG_LEVEL_INFO;
        else{
            std::string ev_level = getenv("LLOG_LV");
            if (ev_level == "WARN")
                m_level_ = LOG_LEVEL_WARNING;
            else if (ev_level == "ERRO")
                m_level_ = LOG_LEVEL_ERROR;
            else
                m_level_ = LOG_LEVEL_INFO;
        }
        console_ = true;
        file_ = false;
    };
public:
	// 单例模式
    static Logger* get_instance() {
        if (m_instance_ == nullptr) {
            mtx_.lock();
            if (m_instance_ == nullptr) {
                m_instance_.reset(new Logger);
            }
            mtx_.unlock();
        }
        return m_instance_.get();
    }

    void init_log_file(std::string log_file_name = "output.log") {
        if (file_ == true)
            log_file.close();
        log_file.open(log_file_name);
        file_ = true;
    }
    
    void set_level(int level) {
        m_level_ = LOG_LEVEL(level);
    }

    int get_level() const {
        return m_level_;
    }

    void set_console(bool b) {
        console_ = b;
    }
    
    int get_console() const {
        return console_;
    }

    void operator+=(const Message& m)
    {
        mtx_.lock();
        if (console_)
            std::cout << m.get_string() + '\n';
        if (file_)
            log_file << m.get_string(false) + '\n';
        mtx_.unlock();
    }

};

std::unique_ptr<Logger> Logger::m_instance_ = nullptr;
std::mutex Logger::mtx_;

# define LLOG(level) if (level >= (*llog::Logger::get_instance()).get_level()) (*llog::Logger::get_instance()) += llog::Message(level, __FILE__, __FUNCTION__, __LINE__)

# define INFO 0
# define WARN 1
# define ERRO 2

# define LLOG_INFO LLOG(INFO)
# define LLOG_WARN LLOG(WARN)
# define LLOG_ERRO LLOG(ERRO)

# define LLOG_LV(level) (*llog::Logger::get_instance()).set_level(level)

# define LLOG_LV_INFO LLOG_LV(INFO)
# define LLOG_LV_WARN LLOG_LV(WARN)
# define LLOG_LV_ERRO LLOG_LV(ERRO)

# define LLOG_FILE(fn) (*llog::Logger::get_instance()).init_log_file(fn);

# define LLOG_CONSOLE_OFF() (*llog::Logger::get_instance()).set_console(false)
# define LLOG_CONSOLE_ON() (*llog::Logger::get_instance()).set_console(true)

} //namespace llog

#endif // _LIGHT_LOG_H
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Cpp(十二) log日志基本使用spdlog
Cpp日志spdlog #1 环境 macOS 10.15.5 spdlog #2 需求分析 日志按等级分到不同的文件 日志按时间分割 #3 使用 #3.1 工程结构 . ├── CMakeLists.txt ├── cmake-build-debug ├── include │ └── spdlog ├── log.hpp └── main.cpp #3.2 CMakeLists.txt cmake_minimum_required(VERSION 3.17) project(my_spdlog)
Autooooooo
2021/03/02
4K0
Cpp(十二) log日志基本使用spdlog
c++日志工具推荐--spdlog
spdlog日志是纯头文件,使用起来比较方便。使用时只需要简单的初始化即可,这里对其初始化做了一个简单的封装,这样使用起来更加方便。 输出到console与输出到文件的级别可以分开设置,并支持动态设置。
fensnote
2021/08/26
9160
spdlog 个人封装,欢迎使用,共同探讨[通俗易懂]
#ifndef _MY_LOG_HPP_ #define _MY_LOG_HPP_
全栈程序员站长
2022/09/10
1.5K0
C++实现的单例模式日志类
在实际生产中,日志是非常重要的调试工具,日志内容至少需要包括时间戳、日志级别、日志内容
叶茂林
2024/05/26
1500
c++日志库实战——spdlog,是不是感觉log4cxx有点笨重,不妨试一试spdlog
在最近新入职同事的推荐下,作者在一个小工具中学习和使用了spdlog,且已发布到线上运行,以下是学习记录。
全栈程序员站长
2022/09/07
1.7K0
c++日志库实战——spdlog,是不是感觉log4cxx有点笨重,不妨试一试spdlog
Linux__之__基于UDP的Socket编程网络通信
本篇博客旨在使用Linux系统接口进行网络通信, 帮助我们更好的熟悉使用socket套接字网络通信, 学会了socket网络通信, 就能发现所谓网络, 不过都是套路而已, 话不多说, 让我们直接进入代码编写部分.
用户11317877
2025/03/24
1220
Linux__之__基于UDP的Socket编程网络通信
C++实现简易log日志系统
在软件开发周期中,不管是前台还是后台,系统一般会采用一个持久化的日志系统来记录运行情况。
恋喵大鲤鱼
2018/08/03
8.5K0
[C++]-日志记录库SPDLog简介[通俗易懂]
spdlog是基于C++11实现的一款纯头文件的日志管理库(git地址:https://github.com/gabime/spdlog,API说明:https://spdlog.docsforge.com/v1.x/1.quickstart/):
全栈程序员站长
2022/09/10
8.7K1
【Linux】别再用printf了!自己造个C++日志库
日志(Log)是程序运行时记录的重要信息,通常用于调试、监控和故障排查。它可以帮助开发者了解程序的运行状态,分析错误,甚至用于安全审计。
用户11305458
2025/03/24
1020
【Linux】别再用printf了!自己造个C++日志库
spdlog使用
参考1:https://www.cnblogs.com/LuckCoder/p/11171609.html
全栈程序员站长
2022/09/10
9180
spdlog 日志库学习,简易封装
spdlog wiki:https://github.com/gabime/spdlog/wiki
全栈程序员站长
2022/09/10
1.3K0
【Linux】日志设计模式与实现
  IT行业这么火, 涌入的人很多. 俗话说林子大了啥鸟都有. 大佬和菜鸡们两极分化的越来越严重. 为了让菜鸡们不太拖大佬的后腿, 于是大佬们针对⼀些经典的常见的场景, 给定了⼀些对应的解决⽅案, 这个就是设计模式。   在IT行业中,设计模式(Design Patterns) 是一套被广泛认可的、用于解决软件设计中常见问题的最佳实践。它们提供了一种标准化的方法来处理特定的设计问题,并且可以帮助开发人员编写更清晰、更具可维护性的代码。
大耳朵土土垚
2025/02/02
790
【Linux】日志设计模式与实现
c++ 日志类 线程安全+缓存
根据上一次的测试,有缓存的日志类性能会更好。用到了time.h类函数,所以在linux下就要改动一下了,windows环境下写的。
forxtz
2020/10/10
1.3K0
开源库推荐——spdlog
在软件开发领域,日志记录是一项至关重要的工作。它不仅是程序运行情况的实时记录者,更是故障排查和系统优化的关键依据。然而,随着项目规模的扩大和复杂性的增加,传统的日志管理方式已经难以满足开发者的需求。这时候,一个高效、灵活的日志库就显得尤为重要。在众多的日志库中spdlog以其出色的性能和便捷的使用方式,赢得了广大开发者的青睐。
程序员的园
2024/07/18
2050
开源库推荐——spdlog
Log:Spdlog初探(1)
Github地址:Github地址 官网介绍:Very fast, header-only/compiled, C++ logging library。轻量,仅有头文件/编译,C++日志库。 注意:Spdlog包含了C++11特性,需使用支持C++11特性的编译器。
全栈程序员站长
2022/09/10
1K0
Log:Spdlog初探(1)
【Log4cpp】项目日志Log4cpp的简单使用
项目日志——Log4cpp 开源的C++类库Log4cpp提供了很多功能,帮助我们的应用程序更方便地记录日志。 ---- 安装与构建 Log4cpp官网——http://log4cpp.sourceforge.net/ wget https://nchc.dl.sourceforge.net/project/log4cpp/log4cpp1.1.x%20%28new%29/log4cpp-1.1/log4cpp-1.1.3.tar.gz tar zxvf log4cpp-1.1.3.tar.gz c
半生瓜的blog
2023/05/13
4200
【Log4cpp】项目日志Log4cpp的简单使用
【NDK】封装日志库
【NDK】封装日志库 0x1需求 供C++、Java调用 控制台输出 文件输出(文件大小) 设置日志等级 0x2 C++ 0x21 LogUtils.h // // Created by 后端码匠 on 2022/11/30. // #ifndef NDKPRACTICE_LOGUTILS_H #define NDKPRACTICE_LOGUTILS_H #include <stdio.h> #include <android/log.h> #include <errno.h> #define
后端码匠
2023/02/27
6660
【NDK】封装日志库
C++日志系统如何设计
笔者在写作本章节的时候,并不敢把此章节的标题叫做《高性能日志系统的设计》,之所以不敢加上“高性能”三个字的原因是,第一,我的对于日志系统设计知识和经验都来自于学习和工作经验,我并不是原创者,只是知识的搬运工;第二,目前有许多优秀的、被广泛使用的开源的日志系统,他们给了我很多启发,不敢在此班门弄斧。不管怎样,笔者还是想写一些自己关于对日志系统的理解和经验,让我们开始吧。
范蠡
2018/12/12
2.6K0
C++日志系统如何设计
【C++】基础:多种日志输出方式(终端、rdbuf、ros、类库)
但重定向后,控制台就不会显示信息了,对于程序调试来说并不友好,因此,可以安装一个UnxUtils,解压后,将/usr/local/wbin添加到环境变量path中,就可以使用如tee一类的命令了。
DevFrank
2024/07/24
7740
【C++】基础:多种日志输出方式(终端、rdbuf、ros、类库)
【Linux】:日志策略 + 线程池(单例模式)
🔥 下面开始,我们结合我们之前所做的所有封装,进行一个线程池的设计。在写之前,我们要做如下准备
IsLand1314
2024/12/20
890
【Linux】:日志策略 + 线程池(单例模式)
相关推荐
Cpp(十二) log日志基本使用spdlog
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验