回车+ctrl+z+回车表示结束的原因

代码如下      
  ---------------------------------------------------------  
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   +   回车可以      
       
  为什么上一种情况不可以?      
  困惑,寻找解答,谢谢!!     


答案:   
 原因分析如下:      
       
  输入缓冲是行缓冲。当从键盘上输入一串字符并按回车后,这些字符会首先被送到输入缓冲区中存储。每当按下回车键后,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   理解为键盘输入的终止,但不是流的终止

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 C 语言中,可以使用 gets 函数来输入多行字符串。gets 函数读取字符串直到遇到换行符或 EOF 为止。 例如: char str[100]; while (gets(str)) { // 在这里处理输入的字符串 } 当输入了 ctrl + z 组合键,gets 函数会返回 NULL,退出循环。但是,由于 gets 函数的安全问题,它已经被废弃了,建议使用 fgets 函数来替代。fgets 函数的用法类似,但是会更加安全。 例如: char str[100]; while (fgets(str, sizeof(str), stdin)) { // 在这里处理输入的字符串 } fgets 函数会从标准输入流 stdin 中读取字符串,读取的字符串包括换行符。当输入了 ctrl + z 组合键并按下回车结束时,fgets 函数会返回 NULL,退出循环。 需要注意的是,fgets 函数在读取到 EOF 时并不会停止,而是会读取空字符串。因此,在循环中需要判断字符串是否为空字符串,才能确定是否输入了 EOF。 例如: char str[100]; while (fgets(str, sizeof(str), stdin)) { if (str[0] == '\0') break; // 在这里处理输入的字符串 } 此时,循环会在输入了 ctrl + z 组合键并按下 ### 回答2: 在C语言中,要输入多行字符串并以"Ctrl+Z"组合键再按一下回车结束输入,可以使用以下方法: 1. 首先,声明一个字符数组来存储输入的字符串,例如`char input[1000];`,这里假设每行字符串的最大长度为1000个字符。 2. 使用`fgets()`函数从标准输入流(stdin)中逐行读取字符串,并将其存储在输入数组中。例如,使用如下循环可以连续读取多行字符串: ```c while (fgets(input, sizeof(input), stdin) != NULL) { // 这里可以处理输入的字符串 } ``` 3. 在上述循环内部,您可以对读取的每行字符串进行操作。例如,您可以打印读取的字符串: ```c while (fgets(input, sizeof(input), stdin) != NULL) { printf("%s", input); // 打印读取的字符串 } ``` 4. 输入过程中,当您希望结束输入时,可以按下"Ctrl+Z"组合键。此时,`fgets()`函数将返回`NULL`,并结束循环。 请注意,`Ctrl+Z`组合键的操作方法可能因操作系统和编译环境而有所不同。在某些操作系统中,您可能需要按下"Ctrl+Z"键并按一下回车键,而在其他系统中,您可能只需按下"Ctrl+D"键即可。 以上就是在C语言中输入多行字符串并通过"Ctrl+Z"组合键再按一下回车结束输入的方法。 ### 回答3: 在C语言中,可以使用`getchar()`函数逐字符读取输入的字符串,直到遇到回车符号。如果需要输入多行字符串并以Ctrl+Z键组合然后再按一下回车结束输入,我们可以使用循环结构来实现。 以下是一个示例代码: ```c #include <stdio.h> int main() { char str[100]; // 声明一个长度为100的字符数组来存储输入的字符串 int i = 0; // 初始化计数器 printf("请输入多行字符串,以Ctrl+Z键组合再按一下回车结束输入:\n"); // 使用循环从标准输入中逐行读取字符串 while (fgets(str, sizeof(str), stdin)) { // 输出读取的字符串 printf("第 %d 行输入的字符串为:%s", i+1, str); i++; // 计数器加1 } printf("\n已结束输入"); return 0; } ``` 在这个示例中, 通过循环使用`fgets()`函数逐行读取输入的字符串,并通过计数器`i`来记录读取的行数。当用户按下Ctrl+Z键组合然后再按一下回车时,循环停止,输出"已结束输入"。 需要注意的是,`fgets()`函数会读取回车符号并包含在输入的字符串中。如果不希望包含回车符号,可以在读取完字符串后,使用字符串处理函数`strtok()`或`strcspn()`函数来去除回车符号。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值