crtl+z、crtl+d、Windows和UNIX/Linux流/檔案結束標誌剖析
以一個簡單明瞭的問題引出咱們探討的主題:
----------------------------------------------------------------------------
//剖析Windows下流/檔案結束標誌CRTL+Z-- 2014/04/17 21:24
#include "stdafx.h"
#include <iostream>
using namespace std;
int main()
{
char ch;
int count= 0;
cin.get(ch);
while(cin.fail() == false
{
cout << ch;
count++;
cin.get(ch);
}
cout << "\n"<< count << " characters read"<< endl;
return 0;
}
--------------------------------------------------------------------------------------------------------------------------------------
問題是:
為什麼輸入一串字元加^z(crtl+z)後,回車,程式不會結束?!
輸入:abcd+^z+回車
輸出:
細心的小夥伴會提出疑問,會不會緊跟zbcd後的^z被“吃掉”了?
這裡我輸入了整整6個^z,是不可能全”吃掉的”,可見問題沒那麼簡單啦>_<|||
程式還是一直等待輸入……耐心看完,解決方法在後面!
1) UNIX/Linux模擬流/檔案結束標誌為CRTL+D,Windows模擬流/檔案結束標誌為CRTL+Z,這裡我們重點討論Windows環境下即CRTL+Z
2) 輸入緩衝是行緩衝。當從鍵盤上輸入一串字元並按回車後,這些字元會首先被送到輸入緩衝區中儲存。每當按下回車鍵後,cin.get() 就會檢測輸入緩衝區中是否有了可讀的資料。cin.get() 還會對鍵盤上是否有作為流結束標誌的 Ctrl+Z 或者 Ctrl+D 鍵按下作出檢查,其檢查的方式有兩種:阻塞式
3) Windows下采用阻塞式檢查 Ctrl+Z、Unix/Linux系統下采用非阻塞式的檢查 Ctrl+D。咱們的問題是在Windows系統下,因此使用阻塞式的 Ctrl+Z 來標識流的結束
4) 非阻塞式檢查指的是按下 Ctrl+D 之後立即響應的方式。如果在按 Ctrl+D 之前已經從鍵盤輸入了字元,則 Ctrl+D的作用就相當於回車,即把這些字元送到輸入緩衝區供讀取使用,此時Ctrl+D不再起流結束符的作用。如果按 Ctrl+D 之前沒有任何鍵盤輸入,則 Ctrl+D 就是流結束的訊號。
5) 阻塞式檢查指的是隻有在回車鍵按下之後才對此前是否有 Ctrl+Z 組合鍵按下進行檢查,這種阻塞式的方式有一個特點:只有按下回車之後才有可能檢測在此之前是否有Ctrl+Z按下。還有一個特點就是:如果輸入緩衝區中有可讀的資料則不會檢測Ctrl+Z(因為有要讀的資料,還不能認為到了流的末尾)。還有一點需要知道:Ctrl+Z產生的不是一個普通的ASCII碼值,也就是說它產生的不是一個字元,所以不會跟其它從鍵盤上輸入的字元一樣能夠存放在輸入緩衝區
※ 從鍵盤上輸入abcd^z 加 回車之後在Windows系統上是這樣處理的:由於回車的作用,前面的 abcd 等字元被送到輸入緩衝區(注意:上面說過了,^z不會產生字元,所以更不會儲存到輸入緩衝區,緩衝區中沒有 ^z 的存在)。這時,cin.get() 檢測到輸入緩衝區中已經有資料存在(因此不再檢查是否有 ^z 的輸入);於是從緩衝中讀取相應的資料。如果都讀取完了,則輸入緩衝區重新變為空,cin.get() 等待新的輸入。可見,儘管有 ^z 按下,但是由於在此之前還有其它輸入字元(abcd),所以流也不會結束。
※ 因此,輸入流結束的條件是:^z之前不能有任何字元輸入(回車除外),否則 ^z 起不到流結束的作用
解決方法:既然緊跟字元的^z沒有進入緩衝區,那我們在字串回顯後立即輸入^z不就得了唄,你對了#^_^#
第一個^z依舊不起作用,第二行”abcd”回顯之後,我們輸入^z,成功退出!