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