本文主要是介绍《C语言杂记》C语言使用gets函数出现的警告问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
首先我们来看个程序。
#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.
虽然能够运行,但作为一个强迫症患者怎么能允许代码有警告出现呢?我们可以从警告的提示可以看到,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;
}
编译运行看看结果。
我们看不到警告了,但是有发先了一个问题,为何有空行,我们并没有输出空行啊,着有事怎么回事呢?
同样是输入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;
}
现在警告没有了,多余的换行也没有了。
总结:
fgets比gets安全,gets没有指定输入字符的大小, fgets会指定大小,如果超出数组大小,会自动根据定义数组的长度截断。
用strlen检测两者的输入的字符串长度,结果不一样。
这篇关于《C语言杂记》C语言使用gets函数出现的警告问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!