zoukankan      html  css  js  c++  java
  • 文件结束符的使用

    代码如下      
    --------------------------------------------------------------      
       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\n";      
       return   0;      
       }      
    ---------------------------------------------------------    
    VC6.0控制台模式下(^z在VC中模拟文件尾)      
    为什么输入一串字符加^z后,回车,程序不会结束??!!      
            
       比如输入:abcd^z       回车      
       程序回显   abcd      
            
       按道理      
       当读到^z时,检测到了文件尾,cin将eofbit,failbit都设为1      
       假如eofbit或failbit设为1,cin.fail则返回true,应该结束程序了,调试中为什么不结束?      
            
       输入:abcd       回车      
       程序回显:abcd       换行符(看不见),然后等待输入,此时按下^Z       回车可以      
            
       为什么上一种情况不可以?      
       困惑,寻找解答,谢谢!!     
      
      
    答案:   
      原因分析如下:      
            
       输入缓冲是行缓冲。当从键盘上输入一串字符并按回车后,这些字符会首先被送到输入缓冲区中存储。每当按下回车键后,cin.get() 就会检测输入缓冲区中是否有了可读的数据。cin.get()还会对键盘上是否有作为流结束标志的 Ctrl+Z或者 Ctrl +D 键按下作出检查,其检查的方式有两种:阻塞式以及非阻塞式。      
            
       阻塞式检查方式指的是只有在回车键按下之后才对此前是否有Ctrl Z组合键按下进行检查,非阻塞式样指的是按下Ctrl  D   之后立即响应的方式。假如在按Ctrl D之前已经从键盘输入了字符,则Ctrl D的作用就相当于回车,即把这些字符送到输入缓冲区供读取使用,此时Ctrl D不再起流结束符的作用。假如按Ctrl D之前没有任何键盘输入,则Ctrl D就是流结束的信号。         
       Windows系统中一般采用阻塞式检查Ctrl Z、Unix/Linux系统下一般采用非阻塞式的检查Ctrl D。楼主是在Windows系统下,因此使用阻塞式的Ctrl Z   来标识流的结束。      
            
       这种阻塞式的方式有一个特点:只有按下回车之后才有可能检测在此之前是否有Ctrl Z按下。还有一个特点就是:假如输入缓冲区中有可读的数据则不会检测Ctrl Z(因为有要读的数据,还不能认为到了流的末尾)。还有一点需要知道:Ctrl Z产生的不是一个普通的ASCII码值,也就是说它产生的不是一个字符,所以不会跟其它从键盘上输入的字符一样能够存放在输入缓冲区。明白了这几点之后就可以来解释楼主提出的问题了。      
            
       从键盘上输入abcd^z加回车之后在Windows系统上是这样处理的:由于回车的作用,前面的abcd等字符被送到输入缓冲区(注意:上面说过了,^z不会产生字符,所以更不会存储到输入缓冲区,缓冲区中没有^z的存在)。这时,cin.get()检测到输入缓冲区中已经有数据存在(因此不再检查是否有^z的输入),于是从缓冲中读取相应的数据。假如都读取完了,则输入缓冲区重新变为空, cin.get()等待新的输入。可见,尽管有^z按下,但是由于在此之前还有其它输入字符(abcd),所以流也不会结束。      
            
       因此,输入流结束的条件就是:^z之前不能有任何字符输入(回车除外),否则^z起不到流结束的作用。   
      
       还有个问题      
       假如输入abcd^zabcd      
       程序回显   abcd      
       并等待输入,也就是^Z后的不再显示了      
            
       -----------------------------------------------      
       假如输入缓冲区中有可读的数据则不会检测Ctrl Z(因为有要读的数据,还不能认为到了流的末尾)。还有一点需要知道:Ctrl Z产生的不是一个普通的ASCII码值,也就是说它产生的不是一个字符,所以不会跟其它从键盘上输入的字符一样能够存放在输入缓冲区。     
      
       假如输入abcd^zabcd      
       程序回显  abcd      
       并等待输入,也就是^Z后的不再显示了      
       为什么后面的却没有处理呢?      
       所以可以把这种情况下的^z理解为键盘输入的终止,但不是流的终止
    --------------------------------------------------------------------------------------------
      
    不过我认为答复仍然没有解决我的疑问。假如当程序提示"please enter the first data"时,直接输入^Z。按照2楼或return0x0朋友给出的方法,由于输入缓冲区中没有可读的数据,则会检测Ctrl Z。可是,实际情况是此时输入的^Z并未终止程序的运行,仍然需要再输一遍^Z才能中止程序。这是为什么呢?

      
    4楼的朋友可能用的是VC2003,正如hai040朋友说的:
      
    vc2003,一次就行  
    before input, current err state is:0  
    please enter the first data  
    ^Z  
    The first data is -858993460  
    after input the first data, current err state is:3  
    Press any key to continue
      
    我用的是VC 6.0  转载:http://lipingke.blog.hexun.com/40431101_d.html

  • 相关阅读:
    关于windows线程的各种状态
    《深入理解计算机系统》(第二版)第二章练习题3
    《深入理解计算机系统》(第二版)第二章中的一练习题2
    Linux进程/线程模型
    用户进程中执行的操作系统
    关于操作系统模型
    《深入理解计算机系统》(第二版)第二章中的一题目
    并发问题互斥(Dekker算法和Peterson算法)
    概率问题:星期二出生的孩子
    Yii添加扩展加载Ckeditor 4.0以上版本
  • 原文地址:https://www.cnblogs.com/myblesh/p/2496522.html
Copyright © 2011-2022 走看看