首页 > 代码库 > gets和fgets函数的区别

gets和fgets函数的区别

1. gets与fgets

  gets函数原型:char*gets(char*buffer);//读取字符到数组:gets(str);str为数组名。

  gets函数功能:从键盘上输入字符,直至接受到换行符或EOF时停止,并将读取的结果存放在buffer指针所指向的字符数组中。

         读取的换行符被转换为null值,做为字符数组的最后一个字符,来结束字符串。

  注意:gets函数由于没有指定输入字符大小,所以会无限读取,一旦输入的字符大于数组长度,就会发生内存越界,

     从而造成程序崩溃或其他数据的错误。

  fgets函数原型:char *fgets(char *s, int n, FILE *stream);//我们平时可以这么使用:fgets(str, sizeof(str), stdin);

          其中str为数组首地址,sizeof(str)为数组大小,stdin表示我们从键盘输入数据。

  fgets函数功能:从文件指针stream中读取字符,存到以s为起始地址的空间里,知道读完N-1个字符,或者读完一行。

  注意:调用fgets函数时,最多只能读入n-1个字符。读入结束后,系统将自动在最后加‘\0‘,并以str作为函数值返回。

2. 细节介绍

  1,上面说到gets函数无限读取,没有上限。但是有些资料介绍说它最多只能读取1024个,所以我写了下面代码测试一下

    (有兴趣的朋友可以测试一下(●‘?‘●))

 1 //测试gets最多读取字符是否超过1024  2 #include <stdio.h> 3 #include <string.h> 4  5 int main() 6 { 7     char str[2048];       //听说gets最多可以读取1024个字符,我们设定一个2048的数组  8     gets(str);            //从键盘输入大于1024个字符 9     int cnt;10     printf("cnt = %d", strlen(str));   //cnt的值就是数组元素个数,是否大于1024呢??? 11     12     return 0;13 }

    经本人亲自测试第一次cnt = 2003,第二次cnt = 2086,第二次程序最后崩溃了,但是不能说明gets的读取上限为2086左右,

    因为程序崩溃是由于内存越界

  2,在来细说一下fgeis的用法,我们以char str[N];fgets(str, N, stdin);为例:

    fgets只能读取N-1个字符,包括最后的‘\n‘,读完结束后系统将自动在最后加‘\0‘gets读完结束后系统自动会将‘\n‘置换成‘\0‘)。

    说到这里就有俩种情况了:

    一:当你从键盘上输入<=N-1个字符(包括‘\n‘)时,那么字符串str会以‘\n\0’结尾。这就造成了strlen(str)比你想象的大 1 ,

      当然你可以通过下面代码将‘\n‘去掉。

1 if(str[strlen(str) - 1] == \n) {      // 去掉换行符2     str[strlen(str) - 1] = \0;   3 }

    二:当你从键盘上输入>N-1个字符(包括‘\n‘)时,那么字符串str会以‘\0‘结尾。

  3,在上面我们提到从键盘输入字符大于N的情况,这时gets和fgets就又有一些区别了,我们通过以下代码来测试一下:

 1 #include <stdio.h> 2 #include <string.h> 3 #define N 5 4  5 int main() 6 { 7     char s1[N]; 8     char s2[N]; 9     fgets(s1, N, stdin);10 //    gets(s1); 11     if(s1[strlen(s1) - 1] == \n) {      // 去掉换行符12         s1[strlen(s1) - 1] = \0;   13     }14 15 //    fflush(stdin);                               //清空缓冲区16     fgets(s2, N, stdin);17 //    gets(s2);18     if(s2[strlen(s2) - 1] == \n) {      // 去掉换行符19         s2[strlen(s2) - 1] = \0;   20     }21 22     printf("%s %s", s1, s2);23     24     return 0;25 } 

    当我们输入12345按下回车,直接就出来1234 5这样的结果。是不是与我们预想的不一样呢?我们单看结果,s1是1234

    s2是5。这是为什么呢?

    这就是fgets与gets的不同之处,我们第一个fgets只读取了1234,将5‘\n‘放入缓冲区中,当程序运行到第二个fgets时,

    它就会直接从缓冲区读取,直到遇到‘\n‘结束所以就会有这样的结果,此时s1是1234‘\0‘,s2是5‘\0‘.我们应该怎么解决这种问题呢?

    我们可以在第二个fgets前加一句fflush(stdin),它是清除缓冲区的作用,大家可以试试,我就不截图了。

    上面我们是用fgets输入的,现在我们换成gets来输入,看看结果吧:

    说到这里就已经不是简单的gets与fgets的问题了,这涉及到了我们从键盘读入的数据在内存中是怎么显示的了,所以我只简单的说一下。

  在内存中s1是这样排序的

 

  ‘\0‘
  6   
  5
  4 
  3 
  2 
  1

    当s2输入进来的时候会变成:

   ‘\0‘
  6
  5
  4  
  3  
   ‘\0‘
  f
  e
  d
  c
  b
  a

    所以输出的时候就会s1输出f,后遇‘\0‘结束,s2输出abcdef,后遇‘\0‘结束。