1. 程式人生 > >crtl+z、crtl+d、Windows和UNIX/Linux流/檔案結束標誌剖析

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+DWindows模擬流/檔案結束標誌為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,成功退出!