首先我们来看个程序。

#include <stdio.h>
#include <string.h>

int main(int argc, char *argv[])
{
char s[100];

printf("输入字符串:\n");
gets(s);

printf("%s\n",s);
return 0;
}

但是在编译的时候回出现如下警告。

main.c: In function ‘main’:
main.c:25:4: warning: ‘gets’ is deprecated (declared at /usr/include/stdio.h:638) [-Wdeprecated-declarations]
gets(s);
^
/tmp/cczUUgFf.o: In function `main':
main.c:(.text+0x30): warning: the `gets' function is dangerous and should not be used.

《C语言杂记》C语言使用gets函数出现的警告问题_fgets

虽然能够运行,但作为一个强迫症患者怎么能允许代码有警告出现呢?我们可以从警告的提示可以看到,gets函数是危险的,不建议去使用。这不就是自相矛盾吗!既然危险,为何又要去写这个API呢!其实啊,写编译器的人和写API的不是一帮人!扯远了,gets函数由于没有指定输入字符的大小,限制输入缓冲区得大小,如果输入的字符大于定义的数组长度,会发生内存越界,堆栈溢出。后果非常严重!

那么有解决办法吗?答案是肯定的,必将强迫症不止我一个,我们可以使用用fgets函数则可以根据定义数组的长度自动截断字符,从而消除一些安全隐患。

fgets(char * s,int size,FILE * stream);

把stream改为stdin 标准输入,就可以从键盘读取东西了。当然对应的也有fputs函数,改成stdout,就可以从屏幕输出东西了。

好了,我们来修改上文的程序。

#include <stdio.h>
#include <string.h>

int main(int argc, char *argv[])
{
char s[100];

printf("输入字符串:\n");
//gets(s);
fgets(s,100,stdin);

printf("%s\n",s);

return 0;
}

编译运行看看结果。

《C语言杂记》C语言使用gets函数出现的警告问题_#include_02

我们看不到警告了,但是有发先了一个问题,为何有空行,我们并没有输出空行啊,着有事怎么回事呢?

同样是输入abc。

gets只有一次换行,这是因为程序的语句printf(“%s\n”,str)

fgets有两次,而第二次是其本身把回车换行符存入了字符串里

所以,gets的长度只有3和输入的字符串长度一样,fgets是4,多出来的是回车换行符。

fgets函数fgets函数用来从文件中读入字符串。fgets函数的调用形式如下:fgets(str,n,fp);此处,fp是文件指针;str是存放在字符串的起始地址;n是一个int类型变量。

函数的功能是从fp所指文件中读入n-1个字符放入str为起始地址的空间内;如果在未读满n-1个字符之时,已读到一个换行符或一个EOF(文件结束标志),则结束本次读操作,读入的字符串中最后包含读到的换行符。

因此,确切地说,调用fgets函数时,最多只能读入n-1个字符。读入结束后,系统将自动在最后加’\0’,并以str作为函数值返回。

gets()将删除新行符, fgets()则保留新行符。

要去掉fgets()最后带的“\0",只要用 s[strlen(s)-1]=’\0’;即可。

fgets不会像gets那样自动地去掉结尾的\n,所以程序中手动将\n位置处的值变为\0,代表输入的结束。

#include <stdio.h>
#include <string.h>

int main(int argc, char *argv[])
{
char s[100];

printf("输入字符串:\n");
//gets(s);
fgets(s,100,stdin);
s[strlen(s)-1]='\0';

printf("%s\n",s);
return 0;
}

现在警告没有了,多余的换行也没有了。

《C语言杂记》C语言使用gets函数出现的警告问题_字符串_03

总结:


 fgets比gets安全,gets没有指定输入字符的大小, fgets会指定大小,如果超出数组大小,会自动根据定义数组的长度截断。



 用strlen检测两者的输入的字符串长度,结果不一样。