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

相关推荐
sul.i6 分钟前
浅析·指针
算法
春日见8 分钟前
策略梯度算法
算法
Brilliantwxx9 分钟前
【算法从零到千】【1-7】 双指针算法
开发语言·c++·笔记·算法·leetcode·推荐算法
小许同学记录成长11 分钟前
孔洞修补算法
算法
随意起个昵称13 分钟前
线性dp-计数类题目9(斐波那契字符串)
算法·动态规划
菜菜的顾清寒18 分钟前
力扣HOT100(49)动态规划 -- 打家劫舍
算法·leetcode·动态规划
Elastic 中国社区官方博客18 分钟前
Hacknight Beijing:基于阿里云与 Elastic 构建 AI Agents
大数据·运维·人工智能·elasticsearch·搜索引擎·阿里云·云计算
葡萄城技术团队21 分钟前
观察生活:人是如何分词的
算法·生活
草莓熊Lotso23 分钟前
【Linux网络】深入理解 HTTP 协议(一):从基础概念到 URL 编码解码
linux·网络·c++·网络协议·http·软件工程
一号弯25 分钟前
用NAVICAT访问非本地服务器的报错问题
运维·服务器