首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 开发语言 > C语言 >

c programming language和EOF的疑问?解决方案

2012-02-08 
c programming language和EOF的疑问?程序如下,是C圣经K&R中的一段,我去掉了一些与我的问题不相关的代码.程

c programming language和EOF的疑问?
程序如下,是C圣经K&R中的一段,我去掉了一些与我的问题不相关的代码.
程序的目的是接收键盘输入并输出长度大于10的字符窜,程序通过第一句红色的语句调用getline函数,执行到第二句红色的语句时我就不太明白了.for语句的条件是输入的字符窜不超过最大长度且字符c不等于EOF且c不等于回车,我输入是zzzzzzzzzzz 回车 EOF ,getline程序会返回main不奇怪,因为我输入回车键后就相当于在输入另一条字符窜,而新的支付窜的第一个字符是EOF所以i=0返回len=0程序结束.问题是当我输入zzzzzzzzzzzzz EOF 时程序依然会结束,也就是说此程序遇到输入EOF就会结束.谁来帮我分析一下啊?是getchar()函数的特性还是EOF在C中的特性导致的呢????


#include<stdio.h>
#define MAXLINE 1000
#define LIMIT 10
int getline(char s[],int lim);
void copy(char to[],char from[]);
main()
{
int len;
int max;
char line[MAXLINE];
char longest[MAXLINE];

max = 0;
while((len = getline(line ,MAXLINE)) > 0 )
  {
  if(len > LIMIT)
  printf("%s",line);

  }

getch();
return 0;
}

int getline(char s[],int lim)
  {
  int c,i;
  for(i = 0; i < lim - 1 && (c = getchar()) != EOF && c != '\n'; ++ i)
  s[i] = c;

  if(c == '\n')
  {
  s[i] = c;
  ++ i;
  }
  s[i] = '\0';
  return i;
  }


[解决办法]
1.EOF作为文件结束符时的情况:

EOF虽然是文件结束符,但并不是在任何情况下输入Ctrl+Z都能够实现文件结束的功能,只有在下列的条件下,才作为文件结束符。
(1)遇到getcahr函数执行时,要输入第一个字符时就直接输入Ctrl+Z,就可以跳出getchar(),去执行程序的其他部分;

(2)在前面有字符输入且不为换行符时,要连着输入两次Ctrl+Z,这时第二次输入的Ctrl+Z起到文件结束符的功能,
其实,这两种情况都可以总结为只有在getchar()提示新的一次输入时,直接输入Ctrl+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 起不到流结束的作用。

网上找的 不知道 有没有用
[解决办法]
输入:
sadfasdfasgasfg^Z
回车,什么都不打印,再回车,打印:
sadfasdfasgasfg
应该是getchar函数把输入的字符串都存到缓冲区
第二次回车再从缓冲区读入数据 
if(c == '\n') 

s[i] = c; 
++ i; 

再加上\n,打印。。。
我是这样想的哈
不过感觉有点不对
望指点
[解决办法]
lz用的是什么编译器?不知道是不是这样:getchar在读取EOF后,并没有清空输入流,而后判断到回车,结束了循环~

试试这段小程序(看看只输入EOF是否就结束了程序):

C/C++ code
#include <stdio.h>int main(){   int c1,c2;   c1=getchar();c2=getchar();   printf("%d,%d\n", c1==EOF, c2=='\n');   return 0;}
[解决办法]
EOF一直也是定义为-1,因为-1不表示任何char,所以可以与char区分,用来做为输入的结束。
输入缓冲是行缓冲。当从键盘上输入一串字符并按回车后,这些字符会首先被送到输入缓冲区中存储。每当按下回车键后,getchar()就会检测输入缓冲区中是否有了可读的数据。

LZ想要什么问题的解答呢
------解决方案--------------------


不同的编译器,getchar() 对^Z处理不同。
不过你的编译器的确非常奇怪,能说一说是什么吗?
我试过VC6, WC, TCC 运行结果都不像你说的那样。

请楼主把main()中的getch(); 去掉,然后在getline()函数中return i;前面加一句:
puts("marked here!");

然后再输入输入zzzzzzzzzzzzz EOF看看是什么结果。

热点排行