区分C语言中getch、getche、fgetc、getc、getchar、fgets、gets函数

时间:2022-04-22 13:31:10

 转: http://blog.csdn.net/digu/article/details/5578821

 

//-------------------------------------------------------------------------------------------------------------

 

 

两个原帖:

           http://www.cnblogs.com/younes/archive/2010/01/05/1639482.html

          http://blog.csdn.net/cxyol/archive/2006/03/18/628324.aspx

 

 

首先,这两个函数不是C标准库中的函数:

int getch(void)    //从标准输入读入一个字符,当你用键盘输入的时候,屏幕不显示你所输入的字符。也就是,不带回显。

int getche(void)    //从标准输入读入一个字符,键盘输入的时候,屏幕显示所输入的字符。带回显。

这两个函数都不带缓冲区,其包含在头文件为conio.h ,需要记住的是conio.h不是C标准库中的头文件。Micorsoft 和 Borland的 C编译器提供了conio.h,用来创建控制台文本用户界面。一般在Windows系统下安装了VS、VC等,就可以包含conio.h头文件。但是一般在Unix、Linux系统中,/usr/include/中都没有这个头文件。

getch和getche在等待用户从键盘输入的时候,用户按下一个键后,不需要按回车,程序自动往下执行。在Linux中,终端输入在缺省情况下是被“一锅端”的,也就是说整行输入是被一起处理的。通常,这是一种人们所希望的方便的办法,但它也意味着在读入数据时必须按一下回车键表示输入行结束后才能得到输入的数据。在游戏中,很多都提供了“老板键”,它的实现,就是利用了这两个函数。

 

 

 

其次,除了getch和getche,其他的都是C标准库中的头文件,包含在头文件stdio.h中。

int fgetc ( FILE * stream ); //从流stream中读一个字符。可以将标准输入stdin作为它的实参,这时候从标准输入读取一个字符。

int getc(FILE * stream);     //和fgetc等效,由fgetc通过宏实现。

int getchar ( void );    //getchar() 函数等待输入直到按回车 才结束(前提是缓冲区没有数据),回车前的 所有输入字符都会逐个显                             //示在屏幕上。但每次调用只读入最开始的一个字符并将其作为函数的返回值

说明:getc、getchar都是通过宏定义借助fgetc实现。如getchar的实现为,#define getchar() fgetc(stdin)。对于getchar函数,因为我们输入的字符串并不是取了第一个字符就把剩下的字符串丢掉了,它还在我们的内存中,就好比,开闸放水,我们把水放到闸里去以后,开一次闸就放掉一点,开一次就放掉一点,直到放光了为止,这里开闸动作就相当于调用一次 getchar() 。我们输入的字符串也是这么一回事,首先我们输入的字符串是放在内存的缓冲区 中的,我们调用一次 getchar() 就把缓冲区中里出口最近的一个字符输出,也就是最前面的一个字符输出,输出后,就把它释放掉了,但后面还有字符串,所以我们就用循环把最前面的一个字符一个个的在内存中释放掉,直到最后把回车符也放空为止,空了之后再在执行 getchar() 就停下等待你的输入了。

注意:

1.在windows环境下,按键盘上的回车产生了 2 个字符 : 回车符 ('/r') 和换行符 ('/n') 。回车符 '/r'(CR:carriage return: 倒车)使光标回到这行的首部,换行符 ('/n')(new line) 然后再换行。但是在linux环境下,按键盘上的回车只产生一个字符'/n'

2.如何清空输入缓冲区的内容?

如果我想让 getchar() 每次都能够等待用户输入的话就要清空缓冲区,下面就介绍方法(不同平台)

C 标准规定 fflush() 函数是用来刷新输出( stdout )缓存的。对于输入( stdin ),它是没有定义的。 但是有些编译器也定义了 fflush(stdin ) 的实现,比如微软的 VC 。其它编译器是否也定义了 fflush(stdin) 的实现应当查找它的手册。 GCC 编译器没有定义它的实现,所以不能使用 fflush(stdin) 来刷新输入缓存。

对于没有定义 fflush(stdin) 的编译器,可以 使用 fgets() 函数来刷新输入缓存 (比用 getchar() 、 scanf() 等函数通用性好)。可以这样忽略输入流中留下的回车等其它输入,从而使下一次的输入总保持一个 “ 干净 ” 的状态。(这个是任何平台下都可以的)
// ...
char sbuf[1024];
// ...
fgets( sbuf, 1024, stdin );
// ...

在 windows 的 vc 下面就可以这样了:

for(int i=0;i<10;++i)

{

       char ch=getchar();

       fflush(stdin); // 每次都会有等待状态了

}

 


fgets()用来从参数stream所指的文件内读入字符并存到参数s所指的内存空间,直到出现换行字符,读到文件尾或是已读了size-1个字符为止,最后会加上NULL作为字符串结束。gets()若成功则返回s指针,返回NULL则表示有错误发生。

char * fgets (char * str, int num, FILE *stream);

                 //从流stream中读入最多num个字符到字符数组str中,当遇到换行符时、或读到num-1个字符时停止。

                 //读入的字符串中最后包含读到的换行符, 自动加上'/0'空字符结尾

char * gets ( char * str ); //从标准输入stdin读取一个字符串,遇到换行或结束时候终止。

                 //不同于fgets,他没有指定num,所以需要注意字符数组str的大小。

                                                                 //读入字符不包括最后的换行符,但自动加上'/0'空字符结尾

说明: fgets和gets之间没有宏定义的关系,彼此各自有自己的实现。蠕虫病毒的实现就是函数gets的“功劳”。gets函数的任务是从流中读入一个字符串。它的调用者会告诉它把读入的字符串放在什么地方。但是,gets()函数并不检查缓冲区大小 ,如果调用者提供了一个指向堆栈的指针,并且get()函数读入的字符数量超过了超过了缓冲区的空间大小,get()会愉快地将多出来的字符继续写入到堆栈中,这就覆盖了堆栈中原来的内容。如:

main()

{

   char line[512];    //在程序的堆栈上分配512个字符的空间

   ...

   gets(line);        //蠕虫病毒的入口,可以将恶意代码通过多出来的数据写入堆栈

}

 

 

 

建议不要用getch和getche,因为它们不是C标准库中的函数。用它们写出的程序可移植性差,不同的编译器不保证可以包含conio.h,其他的函数都是对流操作,都有相关缓冲区,此外建议用fgets函数彻底替代gets函数。

另外,绝大多数的这些get函数,都有对应的put版本。

int fputc ( int character, FILE * stream );

int putc ( int character, FILE * stream );       //通过宏定义和fputc实现

int putchar ( int character );        //通过宏定义实现:#define putchar(c) fputc(c, stdout)

 

int fputs ( const char * str, FILE * stream );

int puts ( const char * str );

说明:两者之间无宏定义实现关系。puts(const char *str)近似等效于fputs(cosnt char *str, stdout),不同点是前者还输出一个'/n'

 

 

 

最后,关于EOF

EOF是在stdio.h文件中定义的符号常量,值为-1。 如:

fputc函数返回一个值:如果输出成功则返回值就是输出的字符;如果输出失败,则返回一个EOF。 

fgetc函数读字符时遇到文件结束符,函数返回一个文件结束标记EOF。如果想从一个磁盘文件顺序读入字符并在屏幕上显示,可以:

ch = fgetc(fp);

while(ch != EOF){

  putchar(ch);

  ch = fgetc(fp);

}

注意,EOF不是可输出字符,因此不能在屏幕上显示。由于ASCII码不可能出现-1,因此EOF定义为-1是合适的。 当读入的字符值等于-1(即 EOF)时,表示读入的已不是正常的字符,而是文件结束符。但以上只适用于读取文本文件的情况。现在ANSI C 已经允许用缓冲文件系统处理二进制文件,而读入某一个字节中的二进制数据的值有可能是-1,而这又恰好是EOF的值。这就出现了需要读入有用数据,却处理为“文件结束”。feof(fp) 用来测试fp所指向的文件当前状态是否是“文件结束” 。如果想顺序读入一个二进制文件数据,可以:

while(!feof(fp)){

   c = fgetc(fp);

   ...

}

 

详情请查阅C标准库。