转载自CSDN博客:http://blog.csdn.net/shenglanya/article/details/52213500
首先要记住的一句话就是Never use gets().
这是因为gets()函数不检查目标数组是否能够容纳输入,而若想把一个字符串读到程序中,最先要做的事情就是预留存储字符串的空间。所以这很容易导致分配的空间不够大而数组越界,然而gets()函数并不检查这个方面,所以导致的结果就是程序很容易出现漏洞,著名的“蠕虫”病毒的原理就是用很长的数据覆盖原有数据导致崩溃。所以对于重要的编程,永远不要使用gets()!
1, gets()的参数是一个地址,因为要把从键盘输入的值确定的放到某一块内存中,所以需要指定它的地址,而通常使用gets(数组名)这种方式来把输入的字符串传入给定的数组中。注意:这个数组的大小一定要事先定义好!若不将数组的大小定义好,就有可能在输入的时候不知道把字符串输入到哪块内存中去了,就有可能会导致对该内存中原代码的覆盖。
2, gets()的第一个用法:直接用gets(array‘s name);这种方式时,由于不知道什么时候才会到字符串结尾,所以每当键入‘\n‘, gets()函数都会自动读取换行符前面的所有内容并且在末尾加上‘\0‘,并且直接把这个字符串返回给调用它的程序,然后gets()再进行读取并且会把读取到的‘\n‘丢弃,这样下一次读取就会在新的一行开始。
例1:
[cpp] view plain copy
- #include <stdio.h>
- #include <stdlib.h>
- #define MAX 81
- int main(void)
- {
- char name[MAX];
- printf("Hi, what‘s your name?\n");
- gets(name);
- printf("Nice name, %s\n", name);
- return 0;
- }
- /*
- Hi, what‘s your name?
- Herry potter
- Nice name, Herry potter
- */
3, gets()的原型: char * gets(char * s)
{
...
return s;
}
所以由此可以看出gets()返回的是一个指向char类型数据的指针,而且这个指针与传递给他的是同一个指针。因此有如下程序:
例2:
[cpp] view plain copy
- #include <stdio.h>
- #include <stdlib.h>
- #define MAX 81
- int main(void)
- {
- char name[MAX];
- char * ptr;
- printf("Hi,what‘s your name?\n");
- ptr = gets(name); // 这里的ptr是char类型的指针,此时ptr指向的是name的首地址。
- printf("%s? Ah? %s!", name, ptr); // 此时输出name和ptr指向的值,可以发现,他们的输出结果都一样。
- return 0;
- }
- /*
- Hi,what‘s your name?
- Herry
- Herry? Ah? Herry!
- */
4, 实际上gets()有两种可能的返回值类型:
1)当程序正常输入字符串时:返回读入字符串的地址,也就是字符串存放的数组的首地址;
2)当程序出现错误或者遇到文件结尾时:返回空指针NULL,注意不要弄混空指针和空字符(‘\0‘);
所以可以很方便的用如下形式检测错误:
while(gets(name) != NULL)
注意:现在基本上不使用gets(),可以说它是一个已经被废弃的函数,现在可以用scanf(), getchar(), fgets()来代替它。