某日一朋友寫了一個HELLO WORLD代碼,出不來結果,代碼如下:
#include <stdio.h>
int
main(int argc, char **argv){
printf("hello world!");
_Exit(0);
}
注意到,在代碼中printf語句打印的字符串最后沒有帶換行符,而且最后調用了_Exit函數,這導致了在終端屏幕上顯示不出來字符串"hello world!"。
首先介紹一下UNIX里面關於標准IO的幾種緩沖機制:
1、全緩沖 。全緩沖指的是系統在填滿標准IO緩沖區之后才進行實際的IO操作;注意,對於駐留在磁盤上的文件來說通常是由標准IO庫實施全緩沖。
2、行緩沖 。在這種情況下,標准IO在輸入和輸出中遇到換行符時執行IO操作;注意,當流涉及終端的時候,通常使用的是行緩沖。
3、無緩沖 。無緩沖指的是標准IO庫不對字符進行緩沖存儲;注意,標准出錯流stderr通常是無緩沖的。
其次介紹一下幾個退出函數:
1、exit ()。調用exit函數之后,它首先會執行一系列的清理處理,包括調用執行各終止處理程序,關閉所有標准IO流等,然后進入內核。
2、_exit ()。與exit不同的是,它不進行清理工作而直接進入內核。此函數由POSIX.1說明,放在unistd.h里面。
3、_Exit ()。同樣,它也不進行清理工作而直接進入內核。此函數跟exit一樣由ISO C說明,放在stdlib.h里面。
現在回過頭來看上面的那段代碼,很容易發現,由於printf函數是行緩沖的(因為它要往終端輸出數據),而且要打印的字符串不帶換行符,因此在它沒有遇到換行符或者沒有填滿緩沖區之前不會進行實際的IO操作,而緊接下來的_Exit函數又立即進入內核沒有處理IO緩沖區,所以我們在終端上看不到hello world語句。
我們可以有很多方法修正這段代碼。最簡單的莫過於增加一個換行符:
#include <stdio.h>
int
main(int argc, char **argv){
printf("hello world!/n");
_Exit(0);
}
此時行緩沖遇到換行符/n,執行實際IO操作。
其次,我們可以調用exit函數,讓它幫我們進行相應的IO處理:
#include <stdio.h>
int
main(int argc, char **argv){
printf("hello world!");
exit(0);
}
exit函數在進入內核之前,對存儲在緩沖區內的數據進行沖洗,然后關閉IO流。
或者,我們可以改變標准輸出流的默認緩沖模式:
#include <stdio.h>
int
main(int argc, char **argv){
setvbuf(stdout, NULL, _IONBF, 0);
printf("hello world!");
_Exit(0);
}
此時,由於調用了setvbuf函數,把標准輸出流默認的行緩沖變成了無緩沖(具體請查閱setvbuf函數實現機制),因此調用printf時立即輸出。
當然,我們還可以調用fclose函數來達到此目的:
#include <stdio.h>
int
main(int argc, char **argv){
printf("hello world!");
fclose(stdout);
_Exit(0);
}
實際上, fclose函數隱含包含了一次fflush操作,把緩沖區內的數據沖洗到終端。