C++开源库glog使用封装--自定义日志输出格式,设置日志保留时间

glog下载和编译

  • glog开源地址

https://github.com/google/glog

  • glog静态库编译
cd /home/wangz/3rdParty/hldglog/glog

mkdir out 
mkdir build && cd build

cmake ..  -DCMAKE_INSTALL_PREFIX=../out -DCMAKE_BUILD_TYPE=Release -DBUILD_SHARED_LIBS=OFF
本文选择的glog版本为glog-0.7.0

Hldglog类封装

#ifndef HLD_GLOG_H
#define HLD_GLOG_H

#define GLOG_USE_GLOG_EXPORT
#include <glog/logging.h>

#define HLD_LOG_INFO LOG(INFO)
#define HLD_LOG_WARNING LOG(WARNING)
#define HLD_LOG_ERROR LOG(ERROR)
// #define LOG_FATAL LOG(FATAL) FATAL消息会终止程序(在记录消息之后),禁用

class Hldglog
{
public:
    static Hldglog *InitGlog(const char *argv, std::string logPath);

private:
    Hldglog() = default;
    virtual ~Hldglog();
    Hldglog(const Hldglog &) = delete;
    Hldglog &operator=(const Hldglog &) = delete;
    Hldglog(Hldglog &&) = delete;
    Hldglog &operator=(Hldglog &&) = delete;

private:
    static void CustomePrefixFormatter(std::ostream &s, const google::LogMessage &m, void *data);

private:
    static std::string m_FilePath;
    static Hldglog *m_instance;
};

#endif

#include "hldglog.h"
#include <iostream>
#include <chrono>
#include <iomanip>
#include <ctime>
#include <functional>

using namespace std::chrono_literals;

std::string Hldglog::m_FilePath = "./log";
Hldglog *Hldglog::m_instance = nullptr;

Hldglog::~Hldglog()
{
    google::ShutdownGoogleLogging();
}

Hldglog *Hldglog::InitGlog(const char *argv, std::string logPath)
{
    if (m_instance == nullptr)
    {
        if (logPath.empty())
        {
            Hldglog::m_FilePath = "./log";
        }
        else
        {
            Hldglog::m_FilePath = logPath;
        }

        google::InitGoogleLogging(argv); // 初始化谷歌的日志库
        // 自定义日志格式
        google::InstallPrefixFormatter(Hldglog::CustomePrefixFormatter, nullptr);

        FLAGS_log_dir = Hldglog::m_FilePath;       // 设置日志文件存放的目录
        FLAGS_minloglevel = 0;                     // 设置日志抑制级别
        FLAGS_stderrthreshold = google::GLOG_INFO; // 设置日志记录到文件的级别
        FLAGS_alsologtostderr = true;              // 错误信息同时输出到终端和文件
        FLAGS_colorlogtostderr = true;             // 设置输出到屏幕的日志显示相应颜色
        FLAGS_max_log_size = 2;                    // 最大日志大小(单位为MB)
        FLAGS_logbufsecs = 0;                      // 缓冲日志输出,默认为30秒,此处改为立即输出(日志实时输出)
        FLAGS_stop_logging_if_full_disk = true;    // 当磁盘被写满时,停止日志输出
        FLAGS_timestamp_in_logfile_name = false;   // 日志文件名取消时间戳

        // 获取当前日期
        time_t now = time(nullptr);
        struct tm *local_time = localtime(&now);

        // 从tm结构体中获取年、月、日
        int year = local_time->tm_year + 1900; // tm_year是以1900年为基的
        int month = local_time->tm_mon + 1;    // tm_mon是以0为1月的
        int day = local_time->tm_mday;         // tm_mday是月份中的哪一天

        std::stringstream ss;
        ss << year << "-" << month << "-" << day;
        std::string current_date = ss.str();

        std::string info_log_path = Hldglog::m_FilePath + "/INFO_" + current_date;
        std::string warn_log_path = Hldglog::m_FilePath + "/WARNING_" + current_date;
        std::string error_log_path = Hldglog::m_FilePath + "/ERROR_" + current_date;

        google::SetLogDestination(google::GLOG_INFO, info_log_path.c_str());    // 设置google::GLOG_INFO级别的日志存储路径和文件名前缀
        google::SetLogDestination(google::GLOG_WARNING, warn_log_path.c_str()); // 设置google::GLOG_WARNING级别的日志存储路径和文件名前缀
        google::SetLogDestination(google::GLOG_ERROR, error_log_path.c_str());  // 设置google::GLOG_ERROR级别的日志存储路径和文件名前缀

        google::SetLogFilenameExtension(".log"); // 设置日志文件的扩展名
        google::EnableLogCleaner(24h * 7);       // 自动删除旧的日志,设置期限为7天

        m_instance = new Hldglog();
    }

    return m_instance;
}

void Hldglog::CustomePrefixFormatter(std::ostream &s, const google::LogMessage &m, void *data)
{
    // [L thread_id] yyyymmdd hh:mm:ss.uuuuuu [file:line]
    // msg...

    // google::GetLogSeverityName(m.severity())[0] 获取日志级别

    s << "[" << google::GetLogSeverityName(m.severity())[0] << ' ' << m.thread_id() << "]"
      << ' '
      << std::setw(4) << 1900 + m.time().year() << "-"
      << std::setw(2) << 1 + m.time().month() << "-"
      << std::setw(2) << m.time().day()
      << ' '
      << std::setw(2) << m.time().hour() << ':'
      << std::setw(2) << m.time().min() << ':'
      << std::setw(2) << m.time().sec() << "."
      << std::setw(6) << m.time().usec()
      << ' '
      << "[" << m.basename() << ':' << m.line() << "]"
      << "\n";
}
  • 使用方法
#include <iostream>
#include "hldglog.h"
#include <unistd.h>

using namespace std;

int main(int argc, char *argv[])
{
    Hldglog::InitGlog(argv[0], "./log");

    HLD_LOG_INFO << "HLD_LOG_INFO";
    HLD_LOG_WARNING << "HLD_LOG_WARNING";
    HLD_LOG_ERROR << "HLD_LOG_ERROR";

    while (true)
    {
        sleep(10);
    }

    return 0;
}
  • 该类设置日志保留的时间为7天

  • 该类的日志输出格式:

    [L thread_id] yyyymmdd hh:mm:ss.uuuuuu [file:line]
    msg…

  • 效果展示

在这里插入图片描述

  • 日志文件的格式
  1. INFO_日期.log
  2. WARNING_日期.log
  3. ERROR_日期.log
std::string info_log_path = Hldglog::m_FilePath + "/INFO_" + current_date;
std::string warn_log_path = Hldglog::m_FilePath + "/WARNING_" + current_date;
std::string error_log_path = Hldglog::m_FilePath + "/ERROR_" + current_date;

google::SetLogDestination(google::GLOG_INFO, info_log_path.c_str());    // 设置google::GLOG_INFO级别的日志存储路径和文件名前缀
google::SetLogDestination(google::GLOG_WARNING, warn_log_path.c_str()); // 设置google::GLOG_WARNING级别的日志存储路径和文件名前缀
google::SetLogDestination(google::GLOG_ERROR, error_log_path.c_str());  // 设置google::GLOG_ERROR级别的日志存储路径和文件名前缀

这样设计的好处:确保了当应用程序在同一日内多次启动时,不会生成多个日志文件,从而有效避免了日志分散的问题,保持日志的连续性和管理的便捷性

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/634450.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

HashMap中添加元素

一、HashMap底层使用了3种结构 hash数组(定位)、链表(存储元素)、红黑树(存储元素,提高查询效率) 二、添加流程描述&#xff1a; 添加元素时&#xff0c;先为元素计算出一个hash值&#xff0c;再用hash值%数组长度得到元素位置&#xff0c;将元素(k:v)封装到Node对象中&…

sql server【 特定分隔符隔开的字符串转表】和【 列转逗号隔开的字符串】

文章目录 引言I 特定分隔符隔开的字符串转表II Sql Server 列转逗号隔开的字符串2.1 多列转行,逗号分隔(字段拼接/字段分割)2.1 案例引言 Sql Server 列转逗号隔开的字符串 和 逆转,常用于数据导出和数据查询。 I 特定分隔符隔开的字符串转表 CREATE FUNCTION [dbo].[GetIDLi…

python科研数据可视化之折线图

例如 &#xff1a; 下面的配色表画出的图很好看。选择喜欢的颜色&#xff0c;找到代码中颜色部分进行修改即可。 代码部分已经有详细的注释&#xff0c;就不一一解释了。另外&#xff0c;如果想要坐标轴从设定的值开始就把下面代码中的范围xlim&#xff0c;ylim进行注释。 imp…

MySQL的备份及恢复

目录 5、MySQL的备份及恢复 5.1 MySQL日志管理 5.1.1 MySQL日志类型 5.1.2 错误日志 5.1.3 通用查询日志 5.1.4 慢查询日志 5.1.5 二进制日志 开启日志 二进制日志管理>又叫日志滚动 二进制日志还原数据 删除二进制日志文件&#xff1a; 5.1.6实例&#xff1a; 使用mysqlbi…

windows远程桌面无法连接,轻松解决 Windows远程桌面无法连接问题的故障排查

Windows远程桌面是一个强大且实用的工具&#xff0c;它允许用户远程访问和操作另一台计算机。然而&#xff0c;有时您可能会遇到无法连接的问题&#xff0c;这无疑会严重影响工作效率和体验。但别担心&#xff0c;本文将为您揭示解决这一问题的关键策略&#xff0c;让您轻松恢复…

2024042701-disjoint-set

并查集 Disjoint-Set 一、前言 并查集的历史 1964年&#xff0c; Bernard A. Galler 和 Michael J. Fischer 首次描述了不相交的并查集&#xff0c;1975 年&#xff0c;Robert Tarjan 是第一个证明O(ma(n))&#xff08;逆阿克曼函数&#xff09;算法时间复杂度的上限&#x…

简易CAD程序:Qt多文档程序的一种实现

注&#xff1a;文中所列代码质量不高&#xff0c;但不影响演示我的思路 实现思路说明 实现DemoApplication 相当于MFC中CWinAppEx的派生类&#xff0c;暂时没加什么功能。 DemoApplication.h #pragma once#include <QtWidgets/QApplication>//相当于MFC中CWinAppEx的派生…

c语言:strcmp

strcmp函数是用于比较两个字符串的库函数&#xff0c;其功能是根据ASCII值逐一对两个字符串进行比较。 语法&#xff1a;strcmp(str1, str2) 返回值&#xff1a; 如果str1等于str2&#xff0c;则返回0。 如果str1小于str2&#xff0c;则返回负数&#xff08;具体值取决于C…

【数据分析面试】51. 读取大型csv文件

题目 假设你是一家科技公司的数据分析师。近期由于管理层变动&#xff0c;新的总经理上任&#xff0c;他想要了解公司过往的交易情况数据&#xff0c;并且这个任务由数据分析团队负责完成。历史交易数据下载导出完成后&#xff0c;团队发现Csv文件大小超过了5个G&#xff0c;使…

解决Wordpress中Cravatar头像无法访问问题

一、什么是Cravatar Gravatar是WordPress母公司Automattic推出的一个公共头像服务&#xff0c;也是WordPress默认的头像服务。但因为长城防火墙的存在&#xff0c;Gravatar在中国时不时就会被墙一下&#xff0c;比如本次从2021年2月一直到8月都是不可访问状态。 在以往的时候&…

OVS名词解释(随手记)

qbr&#xff1a;Linux网桥&#xff0c;为虚拟机提供安全组服务&#xff0c;负责安全隔离&#xff0c;有关安全组的实现。 veth-pair&#xff1a;一对虚拟端口设备 br-int&#xff1a;OVS的核心网桥之一。二三层流量均需经过该网桥&#xff0c;通过local vlan tag实现主机内部不…

Greco:使用ZKP来证明FHE参与方的RLWE密文格式正确

1. 引言 以太坊基金会Enrico Bottazzi 2024年论文Greco: Fast Zero-Knowledge Proofs for Valid FHE RLWE Ciphertexts Formation&#xff0c;开源代码见&#xff1a; https://github.com/privacy-scaling-explorations/greco&#xff08;Rust & Python&#xff09;【基于…

2024最新彩虹聚合DNS管理系统源码v1.3 全开源

简介&#xff1a; 2024最新彩虹聚合DNS管理系统源码v1.3 全开源 聚合DNS管理系统可以实现在一个网站内管理多个平台的域名解析&#xff0c;目前已支持的域名平台有&#xff1a;阿里云、腾讯云、华为云、西部数码、DNSLA、CloudFlare。 本系统支持多用户&#xff0c;每个用户可…

数据库|SQLServer数据库管理系统基本使用

哈喽&#xff0c;你好啊&#xff0c;我是雷工&#xff01; 此处学习是以SQL Server为例。 学习数据服务打开、服务器名称的集中写法、TCP/IP协议的打开和登录模式修改的四个步骤,以下为学习笔记。 数据库管理系统包括&#xff1a;客户端服务端&#xff08;运行在服务器上面的一…

Python编程-后端开发之Django5应用请求处理与模板基础

Python编程-后端开发之Django5应用请求处理与模板基础 最近写项目&#xff0c;刚好用到了Django&#xff0c;现在差不多闲下来&#xff0c;个人觉得单体项目来讲django确实舒服&#xff0c;故写此总结 模板语法了解即可&#xff0c;用到了再看&#xff0c;毕竟分离已经是主流操…

深度学习之基于Yolov3的行人重识别

欢迎大家点赞、收藏、关注、评论啦 &#xff0c;由于篇幅有限&#xff0c;只展示了部分核心代码。 文章目录 一项目简介 二、功能三、系统四. 总结 一项目简介 一、项目背景 行人重识别&#xff08;Person Re-Identification&#xff0c;简称ReID&#xff09;是计算机视觉领域…

AI大模型日报#0523:中国大模型价格战的真相、大模型「上车」、王小川首款 AI 应用

导读&#xff1a;AI大模型日报&#xff0c;爬虫LLM自动生成&#xff0c;一文览尽每日AI大模型要点资讯&#xff01;目前采用“文心一言”&#xff08;ERNIE 4.0&#xff09;、“零一万物”&#xff08;Yi-Large&#xff09;生成了今日要点以及每条资讯的摘要。欢迎阅读&#xf…

Vitis HLS 学习笔记--控制驱动TLP - Dataflow视图

目录 1. 简介 2. 功能特性 2.1 Dataflow Viewer 的功能 2.2 Dataflow 和 Pipeline 的区别 3. 具体演示 4. 总结 1. 简介 Dataflow视图&#xff0c;即数据流查看器。 DATAFLOW优化属于一种动态优化过程&#xff0c;其完整性依赖于与RTL协同仿真的完成。因此&#xff0c;…

指针,指针变量,引用,取地址符,malloce()函数使用,C中“—>” 和“ . ” 作用与区别

目录 一&#xff1a;指针,指针变量&#xff0c;引用&#xff0c;取地址符&#xff1a; 前提 &#xff1a; 1.“ * ” 的两种用途 2." & “的两种用途 2.1&#xff1a;引用 2.2&#xff1a;取地址 补充&#xff1a; 二 : malloc(),动态申请地址空间 1.原型定义…

提权方式及原理汇总

一、Linux提权 1、SUID提权 SUID&#xff08;设置用户ID&#xff09;是赋予文件的一种权限&#xff0c;它会出现在文件拥有者权限的执行位上&#xff0c;具有这种权限的文件会在其执行时&#xff0c;使调用者暂时获得该文件拥有者的权限。 为可执行文件添加suid权限的目的是简…