欢迎访问 生活随笔!

生活随笔

当前位置: 首页 > 编程资源 > 编程问答 >内容正文

编程问答

gets fgets 区别

发布时间:2023/11/30 编程问答 57 豆豆
生活随笔 收集整理的这篇文章主要介绍了 gets fgets 区别 小编觉得挺不错的,现在分享给大家,帮大家做个参考.

http://www.cnblogs.com/aexin/p/3908003.html

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'结束。


总结

以上是生活随笔为你收集整理的gets fgets 区别的全部内容,希望文章能够帮你解决所遇到的问题。

如果觉得生活随笔网站内容还不错,欢迎将生活随笔推荐给好友。