zoukankan      html  css  js  c++  java
  • 刚工作时候学习的一个小知识 规格严格

    楼主ming6()2004-09-15 17:16:03 在 C/C++ / C++ 语言 提问

    代码如下  
      --------------------------------------------------------------  
      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;  
          }  
      ---------------------------------------------------------  
      Vc++   6.0控制台模式下(^z在Vc中模拟文件尾)  
      为什么输入一串字符加^z后,回车,程序不会结束??!!  
       
      比如输入:abcd^z   +   回车  
      程序回显   abcd  
       
      按道理  
      当读到^z时,检测到了文件尾,cin将eofbit,failbit都设为1  
      如果eofbit或failbit设为1,cin.fail则返回true,应该结束程序了,调试中为什么不结束?  
       
      输入:abcd   +   回车  
      程序回显:abcd   +   换行符(看不见),然后等待输入,此时按下^Z   +   回车可以  
       
      为什么上一种情况不可以?  
      困惑,寻找解答,谢谢!!  
      问题点数:50、回复次数:8Top

    1 楼SLSnake(一鸟必杀)回复于 2004-09-15 19:17:44 得分 5

    按道理  
      当读到^z时,检测到了文件尾,cin将eofbit,failbit都设为1  
      如果eofbit或failbit设为1,cin.fail则返回true,应该结束程序了,调试中为什么不结束?  
       
      按你这么说,while   (cin.fail()==false)这里的false,应该改成true啊,当不是文件结尾的时候循环,是文件结尾的时候停。Top

    2 楼whyglinux(山青水秀)回复于 2004-09-15 19:20:22 得分 45

    原因分析如下:  
       
      输入缓冲是行缓冲。当从键盘上输入一串字符并按回车后,这些字符会首先被送到输入缓冲区中存储。每当按下回车键后,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   起不到流结束的作用。Top

    3 楼SLSnake(一鸟必杀)回复于 2004-09-15 19:24:55 得分 0

    楼上的真牛,受教啦Top

    4 楼ming6()回复于 2004-09-16 09:20:10 得分 0

    to   whyglinux(山青水秀)  
       
      还有个问题  
      如果输入abcd^zabcd  
      程序回显   abcd  
      并等待输入,也就是^Z后的不再显示了  
       
      -----------------------------------------------  
      如果输入缓冲区中有可读的数据则不会检测Ctrl+Z(因为有要读的数据,还不能认为到了流的末尾)。还有一点需要知道:Ctrl+Z产生的不是一个普通 的ASCII码值,也就是说它产生的不是一个字符,所以不会跟其它从键盘上输入的字符一样能够存放在输入缓冲区。  
      --------------------------------  
      为什么后面的却没有处理呢?Top

    5 楼whyglinux(山青水秀)回复于 2004-09-16 09:50:26 得分 0

    >>   如果输入abcd^zabcd  
      >>   程序回显   abcd  
      >>   并等待输入,也就是^Z后的不再显示了  
      >>   为什么后面的却没有处理呢?  
       
      所以可以把这种情况下的   ^z   理解为键盘输入的终止,但不是流的终止。Top

    6 楼ming6()回复于 2004-09-16 10:27:31 得分 0

    多谢!Top

    7 楼zhongyj2004(macrosoft)回复于 2004-10-25 14:56:46 得分 0

    感谢   whyglinux(山青水秀)  
      读了很有启发Top

    8 楼zhoujiamurong(有分俺就不要,俺要知识)回复于 2005-03-09 11:39:15 得分 0

    markTop

    相关问题

  • 相关阅读:
    List里如何剔除相同的对象?
    Collections工具类中的sort方法如何比较元素?
    TreeMap和TreeSet在排序时如何比较元素?
    Map的实现类中,哪些是有序的,哪些是无序的,如何保证其有序性?
    LinkedHashMap、LinkedHashSet、LinkedList哪个最适合当作Stack使用?
    ArrayList与LinkedList哪个插入性能高?
    HashSet和HashMap有什么区别?
    HashSet实现原理是什么?有什么特点?
    TreeSet的原理是什么?使用需要注意什么?
    Java中已经数组类型,为什么还要提供集合?
  • 原文地址:https://www.cnblogs.com/diyunpeng/p/2471444.html
Copyright © 2011-2022 走看看