輸入輸出緩沖區和流的概念理解
(本文章關于緩沖區概念的理解大部分取自:http://developer.51cto.com/art/201107/277186.htm)
快遞的寄送過程:
源地址(商家的倉庫)——中轉地(快遞公司的倉庫)——目的地(買家)
我們從淘寶商家買衣服,商家通過快遞公司把商品送到我們手里的這個過程可以形象的解釋下緩存區和流的這個概念。
1, 淘寶商家不會親自把商品給買家送過來的,因為這樣效率太低了,商家會通過快遞公司這個中轉,然后快遞公司再把東西送給買家。淘寶商家就是在鍵盤上打字,買家就是程序,程序需要讀取從鍵盤上的輸入的字,緩沖區就是快遞公司的倉庫。
2, 商品的幾種位置狀態:商家倉庫,快遞倉庫,買家手中,還有一種狀態就是在路上。輸入輸出的流就是指的在路上。
3, 快遞收貨員收到商品就放到自己的中轉倉庫中。但是快遞公司肯定等倉庫中的商品積累到一定程度才開始派送。行緩沖就是遇到換行符時就認為需要執行I/O操作了。
一,緩沖區的概念
緩沖區又可以稱為緩存。計算機中的內存可以被認為是硬盤的緩存。當cpu讀取文件、執行程序時,不會直接從硬盤中讀取,而是先把文件緩存到內存中,然后再從內存中讀取。
對于C++程序來說,當類似cin,getchar這樣的對象或者函數讀取輸入時,不會直接直接讀鍵盤上的輸入,而是這樣的一個過程:cin——輸入緩沖區——鍵盤。我們從鍵盤上輸入的字符先存到緩沖區里面,cin從緩沖區里面讀取輸入。對于輸出來說,程序的結果不會直接顯示到屏幕上,而是先存放到緩沖區,然后cout把內容從緩沖區輸出到屏幕。cin和cout本質上都是對緩沖區中的內容進行操作。
如果沒有緩沖區就會大大降低CPU的效率,因為cpu將不得不一直等待用戶的輸入,而不能執行其他的操作,人打字輸入的速度再快,也比不上CPU的執行速度,人在輸入兩個字符之間的間隔時間,cpu完全可以去干別的事情。
緩沖區分為三種全緩沖、行緩沖和不帶緩沖。
1、全緩沖
在這種情況下,當填滿標準I/O緩存后才進行實際I/O操作。全緩沖的典型代表是對磁盤文件的讀寫。
2、行緩沖
在這種情況下,當在輸入和輸出中遇到換行符時,執行真正的I/O操作。這時,我們輸入的字符先存放在緩沖區,等按下回車鍵換行時才進行實際的I/O操作。典型代表是鍵盤輸入數據。
3、不帶緩沖
也就是不進行緩沖,標準出錯情況stderr是典型代表,這使得出錯信息可以直接盡快地顯示出來。
緩沖區的刷新指的是緩沖區的內容被清空刷新,這也就意味著刷新之前系統會對緩沖區內容進行I/O讀寫。下面4種情況會觸發緩沖區的刷新:
緩沖區滿時;
執行flush語句;
執行endl語句;
關閉文件。
C/C++程序里面的緩沖區指的是為標準輸入與標準輸出設置的緩沖區,如果我們不認為的設置的話,系統會自動的為標準輸入與標準輸入設置一個緩沖區,這個緩沖區的大小通常是4Kb的大小。
ANSI C要求下列緩存特征:
(1) 當且僅當標準輸入和標準輸出并不涉及交互作用設備(鍵盤,屏幕)時,它們才是全緩存的。 讀寫文件的時候就是全緩存。
(2)標準出錯決不會是全緩存的。
(3)標準輸入和輸出涉及交互作用設備時,雖然沒有明確規定是不帶緩存的還是行緩存的,但一般系統默認它們是行緩存的。
因此我們經常要用的標準輸入和輸出,stdin、stdout和stderr的緩存特征是:stdin和stdout是行緩存;而stderr是無緩存的。cin和cout都是從緩沖區讀取
二、流的概念
流是一個過程,一個動態的概念。可以把流想象成水在水管中流動的過程,想象成商品快遞運送的過程。Cin和cout就是執行流這個過程的人。
對于輸入,cin負責把輸入緩沖區中的內容傳遞給程序;
對于輸出,cout負責把輸出緩沖區中的內容傳遞給屏幕。
Cin和cout把緩沖區的數據變成流,然后搬運到相應的目的地。Cin和cout就是個搬運工,搬運的過程就是流。
三、代碼案例
第一段代碼:
{
string str;
int i=0;
while (cin >> str)
{
cout << str<<endl;
cout << ++i << endl;
}
return 0;
}
程序執行過程中輸入:i love you
最終結果是:
i
1
love
2
you
3
執行過程中,程序并不會在每次輸入一個空格時就打印一次,而在在完全輸入一行字符串并摁下回車后,才會打印。原因就是在我們輸入回車之前的一行字符串都只是存放到了為標準輸入分配的緩沖區中,這是一個行緩沖區,在遇到換行符之前,緩沖區不會刷新也就不會觸發I/O操作,cin也就沒有在讀取數據。輸入回車后,cin開始執行I/O操作,讀取緩沖區中的字符:首先讀取i,然后遇到了空格,此次讀取完成,執行循環。然后接著讀取love,又遇到了空格,讀取完成,執行循環。最后讀取了you。
第二段代碼:
{
char c;
//第一次調用getchar()函數
//程序執行時,您可以輸入一串字符并按下回車鍵,按下回車鍵后該函數才返回
c = getchar();
//顯示getchar()函數的返回值
cout << c << endl;
//暫停
system("PAUSE");
//循環多次調用getchar()函數
//將每次調用getchar()函數的返回值顯示出來
//直到遇到回車符才結束
while ((c = getchar()) != '\n')
{
printf("%c", c);
}
//暫停
system("PAUSE");
return 0;
}
執行程序,輸入:abcdefg,然后回車
程序結果如下:
a
bcdefg
第一次執行到getchar時,由于此時緩沖區里面沒有任何內容,所以程序等待鍵盤的輸入,輸入abcdefg后,然后輸入回車,觸發了行緩沖的條件,執行I/O,getchar開始讀取緩沖區的內容,由于此函數只讀取一個字符,所以讀完字符a后,讀取結束,執行下面的語句,將a打印到屏幕。由于緩沖區中字符只被讀取了1個字符a,剩余的bcdefg還在緩沖區中,因此執行到while中的getchar時,直接讀取緩沖區中的內容,也就是依次讀取bcdefg,回車符也是緩沖區中的一個字符,當讀取完回車后,while循環結束。
posted on 2016-05-27 16:10 古月 閱讀(1766) 評論(0) 編輯 收藏 引用 所屬分類: C/C++