当前位置: 首页 > news >正文

【Linux笔记】——简单实习一个日志项目

🔥个人主页🔥:孤寂大仙V
🌈收录专栏🌈:Linux
🌹往期回顾🌹: 【Linux笔记】——线程同步信号量与环形队列生产者消费者模型的实现(PV操作)
🔖流水不争,争的是滔滔不息


  • 一、日志的简介
  • 日志

一、日志的简介

在这里插入图片描述

程序员的日志用于记录开发过程中的关键步骤、决策和问题。通过详细记录,开发者可以追踪代码的演变过程,了解每个功能或模块的实现细节。这种记录有助于在项目后期进行回顾和总结,特别是在需要重构或优化代码时。

日志格式以几个指标是必须得有的
时间戳、日志等级、日志内容
以下几个指标是可选的
文件名行号、进程,线程相关id信息等

现在已经有许多现场的实现日志的方案了。

日志

引入设计模式的概念
设计模式是软件工程中用于解决常见设计问题的可重用解决方案。它们提供了一种标准化的方法来处理特定类型的问题,帮助开发者设计出更灵活、可维护和可扩展的软件系统。设计模式并不是具体的代码,而是描述如何组织代码的模板或蓝图。
下面在代码中聊,模板方法模式

   class Logstrategy // 是基类     {public:~Logstrategy() = default; // 编译器自动生成该析构函数virtual void syncloy(const string &message) = 0;//纯虚继承};class ConsoleLogstrateg : public Logstrategy // 屏幕打印派生类{public:ConsoleLogstrateg(){}void syncloy(const string &message) override //检测继承是否是继承的基类{LockGuard lockguard(_mutex);cout << message << grep;}~ConsoleLogstrateg(){}private:Mutex _mutex;};class FileLogstrateg : public Logstrategy // 指定文件打印的派生类{public:FileLogstrateg(string defaultpath, string defaultfile): _path(defaultpath), _file(defaultfile){LockGuard lockguard(_mutex);if(filesystem ::exists(_path)) // 判断路径存不存在 ,为真不存在retuen,创建新路径{return;}try{filesystem ::create_directories(_path); // 创建新路径}catch (const filesystem ::filesystem_error &e) // 没创建成功捕捉异常{cerr << e.what() << "\n";}}void syncloy(const string &message) override{LockGuard lockguard(_mutex);string filename = _path + _file;  // 文件名ofstream out(filename, ios::app); // 追加写入的方式打开文件if (!out.is_open()){return;}out << message << grep;out.close();}~FileLogstrateg(){}private:string _path;string _file;Mutex _mutex;};

我们的日志想设计两种刷新方法,一种是往控制台中刷新日志,一种是往文件中刷新日志。这里采用模板模式的设计模式。基类就写好模板方法,两个子类一个往控制台中刷一个往文件中刷,两个子类继承模板方法去实现具体的方法。

往控制台中打的子类,在实现打印逻辑的函数中直接用输入输出流打印到控制台就ok了。往文件中打的子类,先判断打印日志存储的路径存不存在,不存在就创建新路径。然后在实现打印逻辑中,创建文件以追加写入的方式打开文件,把内容写入文件中。


Log.hpp

   class Logger    //日志{public:Logger(){EnableFileLogstrateg();}void EnableConsoleLogstrateg()  //选择刷新策略 控制台刷新{_fflush_strategy = make_unique<ConsoleLogstrateg>(); // c++14语法通过指针创建并初始化指针对象}void EnableFileLogstrateg()    //选择刷新策略 文件刷新{_fflush_strategy = make_unique<FileLogstrateg>(defaultpath + "/", defaultfile);}class LogMessage    //内部类。表示一条日志{public:LogMessage(LogLevel& level, string &src_name, int line_number, Logger &logger): _curr_time(GetTimestamp()), _level(level), _pid(getpid()), _src_name(src_name), _line_number(line_number), _logger(logger){stringstream ss;ss << "[" << _curr_time << "]"<< "[" << Leve12str(_level) << "]"<< "[" << _pid << "]"<< "[" << _src_name << "]"<< "[" << _line_number << "]"<< "-";_loginfo = ss.str();    //stringstream流中的信息放到_loginfo中}template <class T>LogMessage &operator<<(const T &info) // 重载{stringstream ss;ss << info;_loginfo += ss.str();return *this;}~LogMessage(){if (_logger._fflush_strategy){_logger._fflush_strategy->syncloy(_loginfo);}}private:string _curr_time;  //时间LogLevel _level;    //等级pid_t _pid;         string _src_name;   //文件int _line_number;   //行号string _loginfo; // 合并后的完整信息Logger &_logger;};//不写&,故意创建临时对象LogMessage operator()(LogLevel level, std::string name, int line) //仿函数调用的时候创建临时对象,对单个日志进行构造{return LogMessage(level, name, line, *this);}~Logger(){}private:unique_ptr<Logstrategy> _fflush_strategy; // 智能指针创建指针};

Logger类是整个日志,LogMessage这个内嵌类是表示一条日志。

      Logger(){EnableFileLogstrateg();}void EnableConsoleLogstrateg()  //选择刷新策略 控制台刷新{_fflush_strategy = make_unique<ConsoleLogstrateg>(); // c++14语法通过指针创建并初始化指针对象}void EnableFileLogstrateg()    //选择刷新策略 文件刷新{_fflush_strategy = make_unique<FileLogstrateg>(defaultpath + "/", defaultfile);}

上面一段代码是在构造的这个日志的时候选择刷新策略。私有成员变量里我们用智能指针创建了一个智能指针对象 _fflush_strategy,两种创新策略方法(控制台刷新,文件刷新)有了智能指针对象就能用make_uniqe实例化对象了。构造日志Logger的时候要选一种默认的构造方法。

class LogMessage    //内部类。表示一条日志{public:LogMessage(LogLevel& level, string &src_name, int line_number, Logger &logger): _curr_time(GetTimestamp()), _level(level), _pid(getpid()), _src_name(src_name), _line_number(line_number), _logger(logger){stringstream ss;ss << "[" << _curr_time << "]"<< "[" << Leve12str(_level) << "]"<< "[" << _pid << "]"<< "[" << _src_name << "]"<< "[" << _line_number << "]"<< "-";_loginfo = ss.str();    //stringstream流中的信息放到_loginfo中}template <class T>LogMessage &operator<<(const T &info) // 重载{stringstream ss;ss << info;_loginfo += ss.str();return *this;}~LogMessage(){if (_logger._fflush_strategy){_logger._fflush_strategy->syncloy(_loginfo);}}private:string _curr_time;  //时间LogLevel _level;    //等级pid_t _pid;         string _src_name;   //文件int _line_number;   //行号string _loginfo; // 合并后的完整信息Logger &_logger;};

这个内嵌类是一条日志,日志要包含,时间、等级、所属文件,行号,合并后的完整信息。构造包含这些信息。

获取时间

    string GetTimestamp() // 获取时间{time_t curr = time(nullptr);struct tm curr_tm;localtime_r(&curr, &curr_tm);char timebuffer[128];snprintf(timebuffer, sizeof(timebuffer), "%4d-%02d-%02d_%02d:%02d:%02d",curr_tm.tm_year + 1900,curr_tm.tm_mon + 1,curr_tm.tm_mday,curr_tm.tm_hour,curr_tm.tm_min,curr_tm.tm_sec);return string(timebuffer);}

其实就是个写入操作,把用时间戳获取的时间写入创建的buffer中,最后返回这个buffer。

错误等级

   enum class LogLevel // 枚举错误类型{DEBUG,INFO,WARNINC,ERROR,FATAL};string Leve12str(const LogLevel &level) // 为避免枚举是整数{switch (level){case LogLevel ::DEBUG:return "DEBUG";case LogLevel ::INFO:return "INFO";case LogLevel ::FATAL:return "FATAL";case LogLevel ::WARNINC:return "WARNINC";case LogLevel ::ERROR:return "ERROR";}}

一个枚举类型,把要写的错误类型进行枚举,为了避免枚举的是整数,所以要写这么个函数,其实可以理解为用switch语句再套一层输出为字符串。
其他信息的构造就不一一阐述了。

 _loginfo = ss.str();    //stringstream流中的信息放到_loginfo中

这里我们把stringstream获取的日志信息都放入_loginfo(合并信息)中

   template <class T>LogMessage &operator<<(const T &info) // 重载{stringstream ss;ss << info;_loginfo += ss.str();return *this;}

为了我们命令写入的信息也要写入合并信息中,这里重载一个函数,<<就把我们自己写入的信息也写入_loginfo(合并信息)中。


下面这里是一个非常妙的设计
RAII+l临时对象的自动析构的经典玩法

LogMessage operator()(LogLevel level, std::string name, int line) //仿函数调用的时候创建临时对象,对单个日志进行构造{return LogMessage(level, name, line, *this);}
//构造LogMessage的时候参数中有一个Logger &logger
LogMessage(LogLevel& level, string &src_name, int line_number, Logger &logger): _curr_time(GetTimestamp()), _level(level), _pid(getpid()), _src_name(src_name), _line_number(line_number), _logger(logger)

这是一个仿函数,创建LogMessage对象的时候,创建的是临时对象返回临时对象。

   ~LogMessage(){if (_logger._fflush_strategy){_logger._fflush_strategy->syncloy(_loginfo);}}

创建的LogMessage析构的时候,这时临时对象刷新完然后析构,让这个日志只活在一行代码内。这时候就不用手动刷新了,临时对象只要语句一结束一析构,就会自动刷新。


  // 全局日志对象Logger logger;// 使用宏,简化用户操作,获取文件名和行号#define LOG(level) logger(level, __FILE__, __LINE__)#define Enable_Console_Log_Strategy() logger.EnableConsoleLogstrateg()#define Enable_File_Log_Strategy() logger.EnableFileLogstrateg()

用宏简化日志系统的调用方式,让使用者写日志时更简单、更自然。

日志源码:源码

相关文章:

  • AI编程辅助哪家强?深度解析主流AI编程工具的现状与未来-优雅草卓伊凡
  • 内核常见面试问题汇总
  • Mujoco 学习系列(二)基础功能与xml使用
  • 鸿蒙开发——7.ArkUI进阶:@BuilderParam装饰器的核心用法与实战解析
  • Oracle中如何解决BUFFER BUSY WAITS
  • Oracle Apps R12——报表入门:如何定义一个Concurrent Program(请求)
  • 【Tauri2】046—— tauri_plugin_clipboard_manager(一)
  • RVTools 官网遭入侵,被用于分发携带 Bumblebee 恶意软件的篡改安装包
  • SUI批量转账几种方法介绍
  • 谈谈对《加密算法》的理解
  • PyTorch中单卡训练、DataParallel(DP)和DistributedDataParallel(DDP)
  • 如何自己建设网站?
  • 第6章 C控制语句:循环
  • Java转Go日记(四十三):Gorm事务
  • 反射在spring boot自动配置的应用
  • HTML应用指南:利用POST请求获取全国申通快递服务网点位置信息
  • 基于Java的校园失物招领系统【附源码】
  • AI预测3D新模型百十个定位预测+胆码预测+去和尾2025年5月20日第83弹
  • Python----循环神经网络(WordEmbedding词嵌入)
  • Java中的ImageIo支持webp解析
  • 凤阳鼓楼脱落瓦片2023年刚经历修复,凤阳县文旅局长回应是否违建等焦点问题
  • 美国公布新型核弹B61-13,威力是广岛原子弹的21倍
  • 热点问答:特朗普与俄乌总统分别通话,他们谈了什么
  • 国家发改委:系统谋划7方面53项配套举措,推动民营经济促进法落地见效
  • 南宁海关辟谣网传“查获600公斤稀土材料”:实为焊锡膏
  • 国家统计局:4月全国规模以上工业增加值同比增长6.1%