Linux 简单日志程序

cpp 复制代码
#include<iostream>
#include<time.h>
#include<stdio.h>
#include<stdlib.h>
#include <stdarg.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 BUFFSIZE 1024
#define LogFile "log.txt"
using namespace std;
class Log
{
public:
    Log()
    {
        printmethod=Classfile;
        path="./log/";
    }
    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 printlog(int level,const string &text)//将日志的内容通过指定的方法打印
    {
        switch(printmethod)
        {
            case Screen://打印到显示器上面
            cout<<text<<endl;
            break;
            case Onefile:
            PrintOneFile(LogFile,text);//打印到一个文件内
            break;
            case Classfile:
            PrintClassFile(level,text);//通过日志等级的不同打印到多个文件中
            break;
            default:
            break;
        }
    }
    void operator()(int level,const char*format,...)
    {
        time_t t=time(NULL);//获取时间戳
        struct tm*ctime=localtime(&t);//将时间戳转为具体的年月日时分秒...存储在结构体
        char leffbuff[BUFFSIZE];
        snprintf(leffbuff,sizeof(leffbuff),"[%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[BUFFSIZE];
        va_list s;//typedef char* va_list 本质上就是一个char*类型
        va_start(s,format);//s指向第一个可变参数的首地址,last是可变参数列表的以知的固定参数,也就是...前面的参数
        vsnprintf(rightbuffer,sizeof(rightbuffer),format,s);//将可变参数列表打印到rightbuff
        va_end(s);//将s置为空
        char buff[BUFFSIZE*2+8];//预留一点空间
        snprintf(buff,sizeof(buff),"%s %s\n",leffbuff,rightbuffer);
        printlog(level,buff);
    }
    void PrintOneFile(const string &logname,const string&text)
    {
        string _logname=path+logname;
        int fd=open(_logname.c_str(),O_WRONLY|O_APPEND|O_CREAT,0666);//以只写,追加打开文件,没有就创建
        if(fd<0)
        {
            cout<<"open failed"<<endl;
            return;
        }
        write(fd,text.c_str(),text.size());//将日志内容写入到文件内
        close(fd);//关闭文件
    }
    void PrintClassFile(int level,const string&text)
    {
        string _logname=LogFile;
        _logname+=".";
        _logname+=LevelToString(level);
        PrintOneFile(_logname,text);
    }
private:
    int printmethod;//打印方式
    string path;//日志存储的文件夹
};

time:返回一个时间戳

struct tm结构体

localtime:可以接收时间戳同时返回一个struct tm结构指针。

snprintf和vsnprintf

cpp 复制代码
int snprintf(char *str, size_t size, const char *format[,argument...]);

str:指向一个字符数组,用于存储格式化后的字符串,该数组的大小至少为 size。

size:指定写入 str 数组中字符的最大个数(包括最后的空字符 '\\0')。

format:包含格式说明符的字符串,它定义了后续参数的输出格式,例: %d,%s

,argument...:可变参数列表,与格式字符串中的格式说明符相匹配。

cpp 复制代码
int vsnprintf (char * sbuf, size_t n, const char * format, va_list arg );

str:指向一个字符数组,用于存储格式化后的字符串,该数组的大小至少为 size。

size:指定写入 str 数组中字符的最大个数(包括最后的空字符 '\\0')。

format:包含格式说明符的字符串,它定义了后续参数的输出格式,例: %d,%s

arg:可变参数列表

可变参数列表

类型va_list本质是一个char*类型,我们要使用可变参数列表,必须首先定义一个va_list类型的变量。

cpp 复制代码
va_list ap;

给va_list类型的变量初始化的函数是va_start函数,初始化以后va_list类型的变量指向第一个可变参数的首地址,该函数的函数原型如下:

cpp 复制代码
void va_start(va_list ap, last);

ap: 这是一个 va_list 类型的对象,是指向可变参数的起始位置指针, last 是最后一个传递给函数的已知的固定参数,即省略号之前的参数。

用来提取可变参数列表中的参数的函数是va_arg,使用一次提取一个,每次提取的参数是直接返回的并且该函数提取的同时会自动将ap指向下一个参数。

cpp 复制代码
type va_arg(va_list ap, type);

销毁va_list类型变量的函数是va_end,其本质就是将指针置为NULL。

cpp 复制代码
void va_end(va_list ap);

ap: 要销毁的变量。

相关推荐
SelectDB1 小时前
Litefuse 开源并推出单进程轻量模式,25 秒就能跑起来的 Agent 可观测与评估平台
运维·后端·自动化运维
kisshyshy2 小时前
🍦 雪糕、食堂、火车厢:三幅漫画吃透栈、队列与链表
javascript·算法
AlfredZhao2 小时前
Docker 容器时区不对,`timedatectl` 不存在怎么办?
linux·timezone
猿人谷9 小时前
不只是 CPU 阈值:STAR 如何用 GAT + Transformer 做容器级自动扩缩容?
人工智能·算法
复杂网络10 小时前
Stable Diffusion 视觉大模型微调技术深度调研
算法
复杂网络10 小时前
基于 Stable Diffusion 架构的视觉大模型代表性工作与原理深度解析
算法
MrZhao40010 小时前
Agent Loop 如何用 Hook 扩展:权限、日志与工具拦截
算法
MrZhao40010 小时前
Agent 为什么需要 Skills:别把所有知识都塞进 system prompt
算法
zzzzzz3101 天前
9K Star 炸裂开源!这个 C 语言写的代码知识图谱,把 Linux 内核索引压缩到了 3 分钟
linux·服务器·sql
XIAOHEZIcode1 天前
Linux系统鼠标偏移常见原因以及修复方案
linux·运维·游戏