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: 要销毁的变量。

相关推荐
我是大咖7 分钟前
二维数组与数组指针
java·数据结构·算法
筵陌16 分钟前
算法:动态规划
算法·动态规划
大江东去浪淘尽千古风流人物18 分钟前
【DSP】xiBoxFilter_3x3_U8 dsp VS cmodel
linux·运维·人工智能·算法·vr
草莓熊Lotso38 分钟前
Python 入门超详细指南:环境搭建 + 核心优势 + 应用场景(零基础友好)
运维·开发语言·人工智能·python·深度学习·学习·pycharm
zhuqiyua39 分钟前
【无标题】
算法
Xの哲學1 小时前
Linux Tasklet 深度剖析: 从设计思想到底层实现
linux·网络·算法·架构·边缘计算
dog2501 小时前
Linux 6.19 TCP 的两个极限拉扯
linux·运维·tcp/ip
Imxyk1 小时前
力扣:1553. 吃掉 N 个橘子的最少天数(记忆化搜索,Dijkstra解法)
算法
新兴AI民工2 小时前
【Linux内核九】进程管理模块:list_head钩子构造双向列表和一些宏定义
linux·运维·list·linux内核
小周学学学2 小时前
ESXI故障处理-重启后数据存储丢失
linux·运维·服务器