1. 程式人生 > >C++:while(cin>>tmp)無法結束

C++:while(cin>>tmp)無法結束

有這樣一段程式碼:

int tmp;
vector<int> arr;
while (cin >> tmp)
{
	arr.push_back(tmp);
}

for (int i = 0; i < arr.size(); i++)
{
	cout << arr[i] << ' ';
}

執行輸入:

結果是無法結束,怎麼回事呢?

cin是終端輸入,也被稱之為標準輸入(standard input),cin的主要功能就是從標準輸入讀入一個值。

當從鍵盤上輸入一串字元並按回車後,這些字元會首先被送到輸入緩衝區中儲存。每當按下回車鍵後,cin.get() 就會檢測輸入緩衝區中是否有了可讀的資料。cin.get() 還會對鍵盤上是否有作為流結束標誌的 Ctrl+Z 或者 Ctrl+D 鍵按下作出檢查。其檢查的方式有兩種:阻塞式

以及非阻塞式

1)阻塞式檢查方式指的是隻有在回車鍵按下之後才對此前是否有 Ctrl+Z 組合鍵按下進行檢查;

2)非阻塞式樣指的是按下 Ctrl+D 之後立即響應的方式。

如果在按 Ctrl+D 之前已經從鍵盤輸入了字元,則 Ctrl+D的作用就相當於回車,即把這些字元送到輸入緩衝區供讀取使用,此時Ctrl+D不再起流結束符的作用。如果按 Ctrl+D 之前沒有任何鍵盤輸入,則 Ctrl+D 就是流結束的訊號。

Windows系統中一般採用阻塞式檢查 Ctrl+Z、Unix/Linux系統下一般採用非阻塞式的檢查 Ctrl+D。這種阻塞式的方式有一個特點:只有按下回車之後才有可能檢測在此之前是否有Ctrl+Z按下。還有一個特點就是:如果輸入緩衝區中有可讀的資料則不會檢測Ctrl+Z(因為有要讀的資料,還不能認為到了流的末尾)。還有一點需要知道:Ctrl+Z產生的不是一個普通的ASCII碼值,也就是說它產生的不是一個字元,所以不會跟其它從鍵盤上輸入的字元一樣能夠存放在輸入緩衝區。

例:從鍵盤上輸入abcd^z 加 回車之後在Windows系統上是這樣處理的:由於回車的作用,前面的 abcd 等字元被送到輸入緩衝區(注意:上面說過了,^z不會產生字元,所以更不會儲存到輸入緩衝區,緩衝區中沒有 ^z 的存在)。這時,cin.get() 檢測到輸入緩衝區中已經有資料存在(因此不再檢查是否有 ^z 的輸入),於是從緩衝中讀取相應的資料。如果都讀取完了,則輸入緩衝區重新變為空,cin.get() 等待新的輸入。可見,儘管有 ^z 按下,但是由於在此之前還有其它輸入字元(abcd),所以流也不會結束。

因此,輸入流結束的條件就是:^z 之前不能有任何字元輸入(回車除外),否則 ^z 起不到流結束的作用。