日志情况分析

本文介绍了如何在C++中使用日志功能,包括不同日志等级的定义,如Info、Debug、Warning、Error和Fatal。同时,展示了如何利用va_list、va_start、va_arg和va_end处理可变参数以及生成时间戳的方法。
摘要由CSDN通过智能技术生成

日志是用来记录信息的,更多是方便于查看各种出错信息。时间、日志等级、日志内容、文件名称已经行号都会显示出来!日志等级是什么?就是当出问题时,根据问题成都列出等级
Info:常规消息
Debug:t调试信息
Waring:报警信息
Error:错误信息,比较严重了可能需要立即处理,不过解决了还是能够执行
Fatal:致命问题,不给你解决时间,会直接结束掉
日志获取时间,
time系统接口
在这里插入图片描述
时间戳设计!
localtime获取时间,年月日,小时分秒
在这里插入图片描述
localtime参数是time的返回值,然后它的参数是一个时间结构体,struct tm*
在这里插入图片描述
里面包含各种属性字段年月日小时分秒
写一个日志程序,snprintf
在这里插入图片描述
是将…的内容按规定格式写入到str数组中并且打印在显示器上
在这里插入图片描述
可变参数形参参数个数不确定,当实参传递多少个参数,形参就会形成几个形参。
可变参数捕捉,要在被调用的函数中取得参数值的话,在C/C++中有专们的函数来处理接收实参,一般情况下需要三个宏和一个类型来配合使用。
类型为:va_list
宏定义:va_start、va_arg、va_end;
使用情况
在这里插入图片描述
有一个具体形参,然后定义va_list类型变量s。va_list:定义了三个宏定义函数va_start,va_arg,va_end所需要的一些基本信息
va_start:表示的是设置头第一个参数,它的第一个参数是va_list 的变量,第二个参数是函数形参的变量名,函数形参只有一个,而这个形参为可变参数中所能接受实参的最大个数!指向可变部分的开头取第一个形参的首地址,然后取地址再+1。s = &n+1指针向后移动n自身类型字节大小,+1跳过一个类型大小
va_arg:表示从被调函数第二个形参开始获取传过来的实参,它的第一个参数是va_list定义的变量,第二个参数是被调函数第二个参数的类型名称,每调用一次va_arg,就获取到一个被调函数中的形参参数
va_end:表示清理va_list定义的变量

1.定义va_list
2.将第一步定义的va_list变量,传入va_start,获取函数参数的第一个
3.反复调用va_arg,获取可变参数函数未定义在形参内的变量
4.用va_end清理va_list,每一次只要定义了va_list类型的变量,都要调用va_end()清理
可变参数必须要有一个参数因为要找到起始地址
而在使用的时候若是实参个数多于了传入的形参个数,多余的是不会计算的,这个参数只是最多能接受的实参个数!

#pragma once

#include <iostream>
#include <time.h>
#include <stdarg.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.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 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 &logname, const std::string &logtxt)
    {
        std::string _logname = path + logname;
        int fd = open(_logname.c_str(), O_WRONLY | O_CREAT | O_APPEND, 0666); // "log.txt"
        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); // "log.txt.Debug/Warning/Fatal"
        printOneFile(filename, logtxt);
    }

    ~Log()
    {
    }

    void operator()(int level, const char *format, ...)
    {
        time_t t = time(NULL);//设置时间戳
        struct tm *ctime = localtime(&t);//localtime可以获取当前时间,它的返回值是一个结构体对象
        char leftbuffer[SIZE];
        //然后可以根据这个对象获取到当前时间,写到一个Buffer数组中
        snprintf(leftbuffer, sizeof(leftbuffer), "[%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);
                    //时间是从1900开始的所有要加1900

        va_list s;
        va_start(s, format);
        char rightbuffer[SIZE];
        vsnprintf(rightbuffer, sizeof(rightbuffer), format, s);
        va_end(s);

        // 格式:默认部分+自定义部分
        char logtxt[SIZE * 2];
        snprintf(logtxt, sizeof(logtxt), "%s %s\n", leftbuffer, rightbuffer);

        // printf("%s", logtxt); // 暂时打印
        printLog(level, logtxt);
    }

private:
    int printMethod;//打印方式
    std::string path;
};
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值