用C語言的時候,您是否還在使用printf函數(shù)來輸出日志呢?您是否考慮過將printf函數(shù)打印的內(nèi)容存到文件中去呢?您是否想擁有一個可選擇的既支持輸出到屏幕又支持存儲到文件中的日志函數(shù)呢?很高興的告訴您,如果您愿意的話,歡迎使用本人編寫的一個一套日志函數(shù),該套函數(shù)由五部分組成,分別是宏變量BUF_SIZE、結(jié)構(gòu)體log_st、log_init函數(shù)、log_debug函數(shù)和log_checksize函數(shù)。其中宏變量BUF_SIZE用來限制每次輸出的日志的最大長度;結(jié)構(gòu)體用來存儲用戶需求,包括文件路徑、文件描述符號、單個文件最大大小、輸出方式標(biāo)志、文件命名標(biāo)志等;log_init函數(shù)用來完成用戶需求錄入、文件創(chuàng)建等功能,在mian函數(shù)的開始調(diào)用一次即可;log_debug函數(shù)的功能跟printf很類似,是在printf基礎(chǔ)上進(jìn)行的擴(kuò)充,實(shí)現(xiàn)將日志輸出到屏幕或者寫入到文件,在需要打印日志的地方調(diào)用該函數(shù);log_checksize函數(shù)用來檢測日志文件大小是否超過最大大小限制,它需要您定時或者定點(diǎn)調(diào)用它,如果一直不調(diào)用,則日志文件將不受指定的最大大小限制。
坊子網(wǎng)站制作公司哪家好,找創(chuàng)新互聯(lián)!從網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、APP開發(fā)、響應(yīng)式網(wǎng)站開發(fā)等網(wǎng)站項(xiàng)目制作,到程序開發(fā),運(yùn)營維護(hù)。創(chuàng)新互聯(lián)2013年開創(chuàng)至今到現(xiàn)在10年的時間,我們擁有了豐富的建站經(jīng)驗(yàn)和運(yùn)維經(jīng)驗(yàn),來保證我們的工作的順利進(jìn)行。專注于網(wǎng)站建設(shè)就選創(chuàng)新互聯(lián)。
一、定義宏變量BUF_SIZE
view plaincopy to clipboardprint?
#define?BUF_SIZE?1024
二、定義log_st結(jié)構(gòu)體
view plaincopy to clipboardprint?
typedef?struct?_log_st?log_st;
struct?_log_st
{
char?path[128];
int?fd;
int?size;
int?level;
int?num;
};
三、定義log_init函數(shù)
參數(shù)說明:path——您要存儲的文件路徑;size——單個文件的最大大小,如果超過該大小則新建新的文件用來存儲;level——日志輸出方式,建議在上層限制其值的范圍為0到3,0表示日志既不輸出到屏幕也不創(chuàng)建文件和保存到文件,1表示日志保存到文件但不輸出到屏幕,2表示日志既輸出到屏幕也保存到文件,3表示日志只輸出到文件而不創(chuàng)建文件和存入文件;num——日志文件命名方式,非0表示以(int)time(NULL)作為文件名來保存文件,文件數(shù)量隨著日志量的遞增而遞增;0表示以“.new”和“.bak”為文件名來保存文件,文件數(shù)量不超過兩個,隨著日志量的遞增,舊的日志文件將被新的覆蓋,更直觀的說就是說.new”和“.bak”文件只保存最近的日志。
view plaincopy to clipboardprint?
log_st?*log_init(char?*path,?int?size,?int?level,?int?num)
{
char?new_path[128]?=?{0};
if?(NULL?==?path?||?0?==?level)?return?NULL;
log_st?*log?=?(log_st?*)malloc(sizeof(log_st));
memset(log,?0,?sizeof(log_st));
if?(level?!=?3)
{
//the?num?use?to?control?file?naming
log-num?=?num;
if(num)
snprintf(new_path,?128,?"%s%d",?path,?(int)time(NULL));
else
snprintf(new_path,?128,?"%s.new",?path);
if(-1?==?(log-fd?=?open(new_path,?O_RDWR|O_APPEND|O_CREAT|O_SYNC,?S_IRUSR|S_IWUSR|S_IROTH)))
{
free(log);
log?=?NULL;
return?NULL;
}
}
strncpy(log-path,?path,?128);
log-size?=?(size??0???size:0);
log-level?=?(level??0???level:0);
return?log;
}
四、定義log_debug函數(shù)
view plaincopy to clipboardprint?
void?log_debug(log_st?*log,?const?char?*msg,?...)
{
va_list?ap;
time_t?now;
char?*pos;
char?_n?=?'\n';
char?message[BUF_SIZE]?=?{0};
int?nMessageLen?=?0;
int?sz;
if(NULL?==?log?||?0?==?log-level)?return;
now?=?time(NULL);
pos?=?ctime(now);
sz?=?strlen(pos);
pos[sz-1]=']';
snprintf(message,?BUF_SIZE,?"[%s?",?pos);
for?(pos?=?message;?*pos;?pos++);
sz?=?pos?-?message;
va_start(ap,?msg);
nMessageLen?=?vsnprintf(pos,?BUF_SIZE?-?sz,?msg,?ap);
va_end(ap);
if?(nMessageLen?=?0)?return;
if?(3?==?log-level)
{
printf("%s\n",?message);
return;
}
if?(2?==?log-level)
printf("%s\n",?message);
write(log-fd,?message,?strlen(message));
write(log-fd,?_n,?1);
fsync(log-fd);
}
五、定義log_checksize函數(shù)
view plaincopy to clipboardprint?
void?log_checksize(log_st?*log)
{
struct?stat?stat_buf;
char?new_path[128]?=?{0};
char?bak_path[128]?=?{0};
if(NULL?==?log?||?3?==?log-level?||?'\0'?==?log-path[0])?return;
memset(stat_buf,?0,?sizeof(struct?stat));
fstat(log-fd,?stat_buf);
if(stat_buf.st_size??log-size)
{
close(log-fd);
if(log-num)
snprintf(new_path,?128,?"%s%d",?log-path,?(int)time(NULL));
else
{
snprintf(bak_path,?128,?"%s.bak",?log-path);
snprintf(new_path,?128,?"%s.new",?log-path);
remove(bak_path);?//delete?the?file?*.bak?first
rename(new_path,?bak_path);?//change?the?name?of?the?file?*.new?to?*.bak
}
//create?a?new?file
log-fd?=?open(new_path,?O_RDWR|O_APPEND|O_CREAT|O_SYNC,?S_IRUSR|S_IWUSR|S_IROTH);
}
}
顯然提問者沒有搞清楚 函數(shù) 聲明和定義 這兩個概念
第一行void abc(void); 是聲明,不是定義,因?yàn)闆]有函數(shù)體,只是告訴Main函數(shù),你要調(diào)用的abc函數(shù)是有的,定義在后面
Main函數(shù)中的abc函數(shù)是調(diào)用,調(diào)用當(dāng)然可以多次使用,函數(shù)的作用也在于此,一次定義,多次調(diào)用,體現(xiàn)代碼復(fù)用的好處
最后void abc(void){ …… } 到結(jié)束,是定義了,說明函數(shù)干什么的
ky 寫 abc函數(shù)里 是推薦的方式,成為局部變量,函數(shù)調(diào)用完后,ky可被系統(tǒng)釋放內(nèi)存
ky不能寫在Main函數(shù)里,abc將無法訪問其他代碼塊(花括號內(nèi))的局部變量
ky可以寫在最外面,即不在任何代碼塊內(nèi),且要寫最頂行(建議),這樣后續(xù)的函數(shù)都可以訪問該(全局)變量,但一般建議不要過多使用全局變量
我們在用C編程時有時需要定義不變的變量,即常量,有兩種方法。
例如:
定義常量PI的兩種方式:
1. #define Pi 3.1415926f;
2. const float pi 3.1415926f;
拓展材料:
常量有整型常量、浮點(diǎn)型常量、字符型常量及字符串常量。 ‘
常量定義是指定義符號常量,用一個標(biāo)識符來代表一個常量,通過宏定義預(yù)處理指令來實(shí)現(xiàn)。
格式:#define標(biāo)識符常量
由用戶命名的標(biāo)識符是符號常量名。作為符號常量名,一般大寫。一旦定義,在程序中凡是出現(xiàn)常量的地方均可用符號常量名來代替。
對使用了符號常量的程序在編譯前會以實(shí)際常量替代符號常量。
全局變量和局部變量的區(qū)別:
用編譯器提供的庫管理工具。
C語言的編譯器都會提供一個命令行工具,可以把自己編譯后的.obj模塊加入指定的庫文件,以后使用時只需要連接該庫文件即可。這個命令行工具通常是lib.exe,用這個工具可以查看庫中的模塊,可以把模塊加入到庫中,可以從庫中刪除模塊。這個工具不僅僅是自己建立的庫文件的管理工具,可以管理所有的庫文件,包括C語言提供的標(biāo)準(zhǔn)庫。