fgets()函数的使用

fgets()函数的使用

虽然用 gets() 时有空格也可以直接输入,但是 gets() 有一个非常大的缺陷,即它不检查预留存储区是否能够容纳实际输入的数据,换句话说,如果输入的字符数目大于数组的长度,gets 无法检测到这个问题,就会发生内存越界,所以编程时建议使用 fgets()

fgets() 的原型为:
1 #include
2 char *fgets(char *c ,int size, FILE *stream) ;

fgets() 虽然比 gets() 安全,但安全是要付出代价的,代价就是它的使用比 gets() 要麻烦一点,有三个参数。它的功能是从 stream 流中读取 size 个字符存储到字符指针变量 s 所指向的内存空间。它的返回值是一个指针,指向字符串中第一个字符的地址。
其中:

①s 代表要保存到的内存空间的首地址,可以是字符数组名,也可以是指向字符数组的字符指针变量名;
②size 代表的是读取字符串的长度;
③stream(!!!!重点理解) 表示从何种流中读取,可以是标准输入流 stdin,也可以是文件流,即从某个文件中读取。标准输入流就是前面讲的输入缓冲区。所以如果是从键盘读取数据的话就是从输入缓冲区中读取数据,即从标准输入流 stdin 中读取数据,所以第三个参数为 stdin。

下面下一个程序:

1 #include
2 int main(void)
3 {
4 char str[20];//定义一个最大长度为19,末尾是’\0’的字符数组来存储字符串
5 printf(“请输入一个字符串:”);
6 fgets(str,7,stdin);//从输入流stdin即输入缓冲区中读取7个字符到字符数组str中
7 printf("%s\n",str);
8 return 0;
9 }

fgets()函数的使用_第1张图片

我们发现输入的是“i love you”,而输出只有“i love”。原因是 fgets() 只指定了读取 7 个字符放到字符数组 str 中。“i love”加上中间的空格和最后的 ‘\0’ 正好是 7 个字符。
那有人会问:“用 fgets() 是不是每次都要去数有多少个字符呢?这样不是很麻烦吗?”不用数! fget() 函数中的 size 如果小于字符串的长度,那么字符串将会被截取;如果 size 大于字符串的长度则多余的部分系统会自动用 ‘\0’ 填充。所以假如你定义的字符数组长度为 n,那么 fgets() 中的 size 就指定为 n–1,留一个给 ‘\0’ 就行了。
!!但是需要注意的是,如果输入的字符串长度没有超过 n–1,那么系统会将最后输入的换行符 ‘\n’ 保存进来,保存的位置是紧跟输入的字符,然后剩余的空间都用 ‘\0’ 填充。所以此时输出该字符串时 printf 中就不需要加换行符 ‘\n’ 了,因为字符串中已经有了。
下面写一个程序看一下:

1 # include
2 int main(void)
3 {
4 char str[30];
5 //char *string = str; //一定要先给指针变量初始化
6 printf(“请输入字符串:”);
7 fgets(str, 29, stdin); //size指定为比字符数组元素少一就行了 写string时前面必须初始化
8 printf("%s", str); //printf中不需要添加’\n’, 因为字符串中已经有了
9 return 0;
10 }

fgets()函数的使用_第2张图片

我们看到,printf 中没有添加换行符 ‘\n’,输出时也自动换行了。
所以 fgets() 和 gets() 一样,最后的回车都会从缓冲区中取出来。只不过 gets() 是取出来丢掉,而 fgets() 是取出来自己留着。但总之缓冲区中是没有回车了!所以与 gets() 一样,在使用 fgets() 的时候,如果后面要从键盘给字符变量赋值,那么同样不需要清空缓冲区。下面写一个程序验证一下。

1 # include
2 int main(void)
3 {
4 char str[30];
5 char ch;
6 printf(“请输入字符串:”);
7 fgets(str, 29, stdin);
8 printf("%s", str); //后面不要加’\n’ 会自动换行
9 scanf("%c", &ch);
10 printf(“ch = %c\n”, ch);
11 return 0;
12 }

fgets()函数的使用_第3张图片
参考博客:https://www.cnblogs.com/Luthais/p/10206576.html

你可能感兴趣的:(C语言知识积累)