【Linux】第三十三站:日志

文章目录

  • 一、实现一个简单的日志
    • 1.简介
    • 2.可变参数
    • 3.错误等级
    • 4.时间
    • 5.打印每一条参数
    • 6.与前面的一些代码搭配使用
  • 二、完整代码

一、实现一个简单的日志

1.简介

我们运行代码的时候,我们希望有各种各样的运行时候的一些信息。这也就是日志

它一半有日志时间,日志的等级,日志的内容,文件的名称和行号。

对于日志等级一半有如下的

Info : 常规消息

Warning : 报警信息

Error : 比较严重了,可能需要立即处理

Fatal : 致命的

Debug : 调试

2.可变参数

关于可变参数我们主要用下面四个中的前三个

image-20240121184554880

我们用如下代码来简单的介绍一下

int sum(int n, ...)
{//char* 可以去提取一个一个的参数va_list s;va_start(s, n);int SUM = 0;for(int i = 0; i < n; i++){SUM += va_arg(s,int);}va_end(s); //s = NULLreturn SUM;
}

image-20240121191257243

在这段代码中,这个va_list其实就是一个char*类型的指针。

我们这个可变参数的函数至少需要带有一个参数。

因为我们传递参数的时候是从右向左传递的,所以我们是可以确定出栈顶的元素就是我们所确定的一个元素的地址

这个va_start是一个宏,代表着让s这个指针指向n。也就是栈顶

然后我们va_arg也是一个宏,它的作用是让s指针往栈里面移动第二个参数的类型大小,并解引用

va_end这个宏的作用是让指针置为空

最终我们就可以计算出这个sum的大小了。这个第一个参数我们一般是用作确定后面的参数有几个

image-20240121191705000

像我们所谓的printf里面也是类似的方法,不过要注意的时候,因为第一个参数是字符串参数,它是可以通过解析字符串里面的%s等格式化控制来确定后面有几个参数的,从而完成依次的替换的。

3.错误等级

我们现在来设计我们的日志的函数

只需要下面这样即可

void logmessage(int level, char* format, ...)
{}

首先对于等级,我们可以直接用宏来表示

#define Info 0
#define Debug 1
#define Warning 2
#define Error 3
#define Fatal 4

4.时间

接下是时间,我们可以使用时间戳,不过其实还有其他的系统调用接口

image-20240121193700415

#include <sys/time.h>
int gettimeofday(struct timeval *tv, struct timezone *tz);

第一个参数是一个结构体,里面的两个参数它可以获取秒,微秒。

第二个参数与时区相关,我们可以不用管,直接置为NULL即可

还有一个接口是这样的

#include <time.h>
struct tm *localtime(const time_t *timep);

image-20240121194806434

这个函数的功能是将时间戳转化为一个结构体,这个结构体里面有时分秒月年等信息

我们可以简单的应用一下

image-20240121200140083

我们会注意到时间存在一定的问题,这是因为它是从1900年开始计时的,这个月是从0开始的,所以我们可以进行一下调整

image-20240121200322663

如下所示就正确了

void logmessage(int level, char* format, ...)
{time_t t = time(nullptr);struct tm* ctime = localtime(&t);printf("%d-%d-%d %d:%d:%d\n",ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday, ctime->tm_hour, ctime->tm_min, ctime->tm_sec);
}

image-20240121200416569

5.打印每一条参数

我们可以用vsnprintf函数

image-20240121204556637

最终我们的代码为如下

void logmessage(int level, char* format, ...)
{char leftbuffer[SIZE];time_t t = time(nullptr);struct tm* ctime = localtime(&t);snprintf(leftbuffer, SIZE,"[%s][%d-%d-%d %d:%d:%d]", levelToString(level).c_str(), ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday, ctime->tm_hour, ctime->tm_min, ctime->tm_sec);char rightbuffer[SIZE];va_list s;va_start(s, format);vsnprintf(rightbuffer, SIZE, format, s);va_end(s);char logtxt[2*SIZE];snprintf(logtxt, sizeof(logtxt), "%s %s\n", leftbuffer, rightbuffer);std::cout << logtxt << std::endl;
}  

image-20240121210712783

6.与前面的一些代码搭配使用

如下所示,这里我们就引进了我们刚刚使用的日志

image-20240121211321574

运行结果如下,我们现在可以看到,这样子就规范多了

image-20240121211442163

二、完整代码

如下代码所示,我们再将前面的日志给封装成一个类,然后实现一下分类打印,向文件打印,向显示屏打印等等功能。

#pragma once#include <stdarg.h>
#include <iostream>
#include <time.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>#define SIZE 1024#define Info 0
#define Debug 1
#define Warning 2
#define Error 3
#define Fatal 4#define Screen    1
#define Onefile   2
#define Classfile 3#define LogFile "log.txt"class Log
{
public:Log(){printMethod = Screen;path = "./log/";}void Enable(int method){printMethod = method;}std::string levelToString(int level){switch(level){case Info : return "Info";case Debug : return "Debug";case Warning : return "Warning";case Error : return "Error";case Fatal : return "Fatal";default : return "None";}}void operator()(int level, char* format, ...){char leftbuffer[SIZE];time_t t = time(nullptr);struct tm* ctime = localtime(&t);snprintf(leftbuffer, SIZE,"[%s][%d-%d-%d %d:%d:%d]", levelToString(level).c_str(), ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday, ctime->tm_hour, ctime->tm_min, ctime->tm_sec);char rightbuffer[SIZE];va_list s;va_start(s, format);vsnprintf(rightbuffer, SIZE, format, s);va_end(s);char logtxt[2*SIZE];snprintf(logtxt, sizeof(logtxt), "%s %s\n", leftbuffer, rightbuffer);//std::cout << logtxt << std::endl;PrintLog(level, logtxt);}   // void logmessage(int level, char* format, ...)// {//     char leftbuffer[SIZE];//     time_t t = time(nullptr);//     struct tm* ctime = localtime(&t);//     snprintf(leftbuffer, SIZE,//         "[%s][%d-%d-%d %d:%d:%d]", levelToString(level).c_str(), //         ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday, ctime->tm_hour, ctime->tm_min, ctime->tm_sec);//     char rightbuffer[SIZE];//     va_list s;//     va_start(s, format);//     vsnprintf(rightbuffer, SIZE, format, s);//     va_end(s);//     char logtxt[2*SIZE];//     snprintf(logtxt, sizeof(logtxt), "%s %s\n", leftbuffer, rightbuffer);//     //std::cout << logtxt << std::endl;//     PrintLog(level, logtxt);// }   void PrintLog(int level, const std::string& logtxt){switch(printMethod){case Screen:std::cout << logtxt << std::endl;break;case Onefile:PrintOnefile(LogFile , logtxt);break;case Classfile:PrintClassfile(level, logtxt);break;default: break;}}void PrintOnefile(const std::string& filename, const std::string& logtxt){std::string logname = path + filename;int fd = open(logname.c_str(), O_WRONLY|O_CREAT|O_APPEND, 0666);if(fd < 0){return;}write(fd, logtxt.c_str(), logtxt.size());close(fd);}void PrintClassfile(int level, const std::string& logtxt){std::string filename = LogFile;filename += ".";filename += levelToString(level);PrintOnefile(filename, logtxt);}~Log(){}
private:int printMethod;std::string path;
};

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

亚信安慧AntDB全链路实时化的未来之选

近年来&#xff0c;随着数字化时代的到来&#xff0c;企业对实时数据的需求日益增长。在复杂的商业环境中&#xff0c;企业需要实时了解市场变化、实时响应客户需求&#xff0c;以保持竞争力。为了满足这一需求&#xff0c;亚信安慧AntDB应运而生&#xff0c;成为企业实时数据处…

Python 算法交易实验67 第一次迭代总结

说明 在这里对第一次迭代&#xff08;2023.7~ 2024.1&#xff09;进行一些回顾和总结&#xff1a; 回顾&#xff1a; 1 实现了0~1的变化2 在信息隔绝的条件下&#xff0c;无控制的操作&#xff0c;导致被套 总结&#xff1a; 思路可行&#xff0c;在春暖花开的时候&#x…

【5G 接口协议】N2接口协议NGAP(NG Application Protocol)介绍

博主未授权任何人或组织机构转载博主任何原创文章&#xff0c;感谢各位对原创的支持&#xff01; 博主链接 本人就职于国际知名终端厂商&#xff0c;负责modem芯片研发。 在5G早期负责终端数据业务层、核心网相关的开发工作&#xff0c;目前牵头6G算力网络技术标准研究。 博客…

AI企业知识库:企业智慧之源,引领未来发展

AI企业知识库作为优秘智能的核心技术&#xff08;shuziren06&#xff09;&#xff0c;具有许多引人注目的功能和特点。接下来让我们深入探讨一下AI企业知识库如何在实际应用中助力企业。以下是对AI企业知识库的更详细介绍&#xff1a; 1. 强大的知识整合能力 AI企业知识库能够…

如何搭建MariaDB并实现无公网ip环境远程连接本地数据库

&#x1f308;个人主页: Aileen_0v0 &#x1f525;热门专栏: 华为鸿蒙系统学习|计算机网络|数据结构与算法 ​&#x1f4ab;个人格言:“没有罗马,那就自己创造罗马~” 文章目录 1. 配置MariaDB数据库1.1 安装MariaDB数据库1.2 测试局域网内远程连接 2. 内网穿透2.1 创建隧道映射…

找不到msvcr100dll或msvcr100dll丢失怎么办,5种靠谱的解决方法分享

MSVCR100.dll文件的丢失可能会引发一系列系统运行和应用程序功能上的问题。作为Microsoft Visual C运行库中的一个关键动态链接库文件&#xff0c;它的缺失会导致依赖于此文件的软件无法正常启动或执行预期功能。具体表现可能包括但不限于&#xff1a;应用程序崩溃、闪退&#…

XHCMS靶场小记(熊海)

文件包含漏洞 通过url判断文件包含漏洞&#xff08;参数直接引入文件&#xff09; 发现文件包含漏洞可以先尝试php伪协议进行命令执行&#xff08;php://input或者data://text/plain&#xff09;; 发现allow_url_include参数应该没开启&#xff1b;那么寻找文件上传点上传图…

如何本地安装Python Flask并结合内网穿透实现远程开发

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

冒泡排序-BubbleSort

1、基本思路 从数组的左边开始&#xff0c;比较两个元素的大小&#xff0c;当左边大于右边时&#xff0c;更换左右元素位置&#xff0c;否则不改变&#xff1b;接着向右移动一步&#xff0c;比较第二个元素和第三个元素的大小&#xff0c;重复上述操作&#xff0c;直到最后一个…

ELK 分离式日志(1)

目录 一.ELK组件 ElasticSearch&#xff1a; Kiabana&#xff1a; Logstash&#xff1a; 可以添加的其它组件&#xff1a; ELK 的工作原理&#xff1a; 二.部署ELK 节点都设置Java环境: 每台都可以部署 Elasticsearch 软件&#xff1a; 修改elasticsearch主配置文件&…

<信息安全>《1 国内主要企业网络安全公司概览(一)》

1 深信服科技股份有限公司 信息内容LOGO成立日期2000年12月25日成立。总部深圳市南山区学苑大道1001号南山智园A1栋是否上市深信服[300454]A股市值265亿主要产品企业级网络安全云计算IT基础设施数据通信物联网员工规模9000人分支机构全球50多个荣誉国家级高新技术企业、中国软…

基于深度学习的细胞感染性识别与判定

基于深度学习的细胞感染性识别与判定 基于深度学习的细胞感染性识别与判定引言项目背景项目意义项目实施数据采集与预处理模型选择与训练模型评估与优化 结果与展望结论 基于深度学习的细胞感染性识别与判定 引言 随着深度学习技术的不断发展&#xff0c;其在医学图像处理领域…