您的位置:首页 > 其它

gets和fgets函数的区别

2017-01-16 16:02 183 查看


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
;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
;
8     char s2
;
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'结束。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  C