(轉)標准I/O緩沖:全緩沖、行緩沖、無緩沖 .


某日一朋友寫了一個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操作,把緩沖區內的數據沖洗到終端。

 http://blog.csdn.net/seton040/article/details/4504825


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM