《C语言杂记》C语言使用gets函数出现的警告问题
Posted Bruceoxl
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了《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函数出现的警告问题的主要内容,如果未能解决你的问题,请参考以下文章
《C语言杂记》C语言异常处理之 setjmp()和longjmp()
《C语言杂记》解决 error c4996 ‘fopen‘ This function or variable may be unsafe