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

fgets() 的原型为:

# include <stdio.h>
char *fgets(char *s, int size, FILE *stream);

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

其中:s 代表要保存到的内存空间的首地址,可以是字符数组名,也可以是指向字符数组的字符指针变量名。size 代表的是读取字符串的长度。stream 表示从何种流中读取,可以是标准输入流 stdin,也可以是文件流,即从某个文件中读取,这个在后面讲文件的时候再详细介绍。标准输入流就是前面讲的输入缓冲区。所以如果是从键盘读取数据的话就是从输入缓冲区中读取数据,即从标准输入流 stdin 中读取数据,所以第三个参数为 stdin。

下面写一个程序:


  1. # include <stdio.h>
  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. }

输出结果是:
请输入一个字符串:i love you
i love

我们发现输入的是“i love you”,而输出只有“i love”。原因是 fgets() 只指定了读取 7 个字符放到字符数组 str 中。“i love”加上中间的空格和最后的 '\0' 正好是 7 个字符。

那有人会问:“用 fgets() 是不是每次都要去数有多少个字符呢?这样不是很麻烦吗?”不用数!fget() 函数中的 size 如果小于字符串的长度,那么字符串将会被截取;如果 size 大于字符串的长度则多余的部分系统会自动用 '\0' 填充。所以假如你定义的字符数组长度为 n,那么 fgets() 中的 size 就指定为 n–1(这里貌似有问题),留一个给 '\0' 就行了。

以上是从c语言中文网复制过来的,但是看下面的例子:

#include <stdio.h>
int main()
{
char arr[5];
char *p = arr;
//按照上面的叙述,这里4赋值给fget中间的参数size
fgets(p,4,stdin);
for(int i = 0;i != 5; ++ i){if(arr[i] == '\0')printf("\\0");else if(arr[i] == '\n')printf("\\n");elseprintf("%c",arr[i]);}return 0;
}

按照上面的说法, 输入abcd后,输出结果却是:abc\0

abcd
abc\0

      结果发现,赋值4后,编译器直接把输入字符串当成长度是3字符串了(算上结尾的\0是4个),这和我们的预期是不同的,我们分配的字符数组长度是5的,当然希望使用总长度5个字符空间。所以,如果用fgets为字符数组获取值,fgets的中间size参数直接赋值字符数组长度即可,不必是n-1。(如果有错,希望各位指正)

但是需要注意的是,如果输入的字符串长度没有超过 n–1,那么系统会将最后输入的换行符 '\n' 保存进来,保存的位置是紧跟输入的字符,然后剩余的空间都用 '\0' 填充。所以此时输出该字符串时 printf 中就不需要加换行符 '\n' 了,因为字符串中已经有了。

下面写一个程序看一下:


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

输出结果是:
请输入字符串:i love studying C语言
i love studying C语言

我们看到,printf 中没有添加换行符 '\n',输出时也自动换行了。

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


  1. # include <stdio.h>
  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. }

输出结果是:
请输入字符串:i love you
i love you
Y
ch = Y

反馈:

1.sizeof()作用在字符串上不记结尾\0。也就是返回除去结尾\0后的长度。

2.sizeof()作用在数组上是数组的实际长度,无论数组里面元素是否赋值,结果都是数组长度。

3.fgets()给字符数组赋值的时候,会把末尾的回车键也放入到字符数组中(前提是有地方放),字符数组最后一个字符一定是\0补充。例如:

#include <stdio.h>
int main()
{
char arr[5];
char *p = arr;
fgets(p,5,stdin);
for(int i = 0;i != 5; ++ i){if(arr[i] == '\0')printf("\\0");else if(arr[i] == '\n')printf("\\n");elseprintf("%c",arr[i]);}return 0;
}
a
a\n\0\0\0
ab
ab\n\0\0
abc
abc\n\0
abcd
abcd\0
abcde
abcd\0

4.字符数组实际接受输入字符长度是:字符数组长度- 1。如果输入字符串长度大于

字符数组长度 - 1,那么其余部分不会丢弃,会继续留在内存缓存中。如下:

#include <stdio.h>
int main()
{
char arr[5];
char *p = arr;
fgets(p,5,stdin);
for(int i = 0;i != 5; ++ i){if(arr[i] == '\0')printf("\\0");else if(arr[i] == '\n')printf("\\n");elseprintf("%c",arr[i]);}
char ch;
scanf("%c",&ch);
printf("%c",ch);return 0;
}

第一行是输入,第二行是输出,可以看到,程序中即使有2个接收输入的语句(分别是fgets和scanf)语句,但是第一次输入abcdefghijklmn后,没等第二次输入后,程序已经运行出结果了,因为fgets只能读取5-1=4个字符,从e一直到最后一个字符在第一次输入后就留在缓存中了,程序运行到scanf函数的时候,scanf发现从缓存中可以读取字符,所以不提示输入立刻就读取了e,然后输出结果。

abcdefghijklmn
abcd\0er

再次,其实只要程序输入字符数<= 4个,scanf()函数都不会有提示,为何?比如输入abcd后,那么接着用户会输入\n,所以实际上已经输入了5个字符,fgets会接受前面4个,最后面的\n留在缓存中了,等到scanf运行时候,会直接读取\n。所以此时scanf也不会提示。准确点来说,第一次输入字符数<=3个,scanf就会提示,否则不会提示。

无论哪个输入流,都不会忽略\n,哪怕是scanf也是如此。

例如:

scanf("%c",&ch1);
scanf("%c",&ch2);
printf("%c",ch1);
printf("%c\n",ch2);

如果这段代码输入2个回车,程序立刻会结束,输入2个回车。因为回车也是字符。

fgets函数及其用法,C语言fgets函数详解相关推荐

  1. function函数的用法c语言,function函数

    function()函数的具体用法 请大侠具体告知,高分赠送. function()函数语法: function 函数名(参数1 [参数2]) { 函数体 return 返回值 } 举例: javas ...

  2. 帧栈使用的基本用法c语言,栈帧详解

    一. 理解栈帧 栈帧是什么,我们基本的理解是栈帧是栈帧也叫过程 活动记录,是 编译器用来实现过程/ 函数调用的一种数据结构.通俗来说栈帧就时C语言函数在调用的过程中的调用原理,就是当我们执行一个函数操 ...

  3. c语言中各种函数的作用,C语言常用函数用法大全

    C语言是当中广泛的计算机编程语言,是所有计算机编程语言的祖先,其他计算机编程语言包括当前流行的Java语言,都是用C语言实现的,C语言是编程效率高的计算机语言,既能完成上层应用开发,也能完成底层硬件驱 ...

  4. python中trunc函数_Oracle trunc()函数的用法及四舍五入 round函数

    --Oracle trunc()函数的用法 /**************日期********************/ 1.select trunc(sysdate) from dual  --20 ...

  5. python中divmod函数的用法_Python中divmod函数的用法

    Python中divmod函数的用法,语言,余数,是一种,面向对象,函数 Python中divmod函数的用法 Python中divmod函数的用法 在Python中divmod函数的作用是把除数和余 ...

  6. python装饰器函数-Python函数装饰器常见使用方法实例详解

    本文实例讲述了Python函数装饰器常见使用方法.分享给大家供大家参考,具体如下: 一.装饰器 首先,我们要了解到什么是开放封闭式原则? 软件一旦上线后,对修改源代码是封闭的,对功能的扩张是开放的,所 ...

  7. python跨函数调用变量_对python中不同模块(函数、类、变量)的调用详解

    首先,先介绍两种引入模块的方法. 法一:将整个文件引入 import 文件名 文件名.函数名( ) / 文件名.类名 通过这个方法可以运行另外一个文件里的函数 法二:只引入某个文件中一个类/函数/变量 ...

  8. python中search和match的区别_Python中正则表达式match()、search()函数及match()和search()的区别详解...

    match()和search()都是python中的正则匹配函数,那这两个函数有何区别呢? match()函数只检测RE是不是在string的开始位置匹配, search()会扫描整个string查找 ...

  9. python函数定义及调用-python函数声明和调用定义及原理详解

    这篇文章主要介绍了python函数声明和调用定义及原理详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 函数是指代码片段,可以重复调用,比如我们前 ...

  10. python函数声明和调用定义及原理详解

    这篇文章主要介绍了python函数声明和调用定义及原理详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 函数是指代码片段,可以重复调用,比如我们前 ...

最新文章

  1. 从做大牛那里整理的Python函数相关的学习笔记,希望对你有帮助
  2. 【原创视频教程】学生信息管理系统6--学员信息管理(完结篇)
  3. android 框架LoonAndroid,码农偷懒专用(2014/8/6更新)
  4. java 向上抛异常_java throws 向上抛出的概念问题
  5. 【渝粤教育】国家开放大学2018年春季 8661-21T传感与检测技术 参考试题
  6. vivox7刷linux系统,Vivo 找来宋仲基帮你送 X7
  7. android中的适配器模式
  8. Python菜鸟快乐游戏编程_pygame
  9. (已成功)windows下,VS2012+Qt5.5.1的安装、路径配置、项目配置(其它版本可参考)
  10. 关于产品MVP的定义与实践
  11. java自由职业者_自由职业者的7个重要技巧
  12. 单元测试用例 php,PHP 单元测试(PHPUnit)(2)
  13. 干货▍全网通俗易懂的数据竞赛大杀器XGBoost 算法详解
  14. 实战项目一、安居客(北京) 二手房抓取房源信息
  15. 计算机毕业设计JavaVue框架电商后台管理系统(源码+系统+mysql数据库+lw文档)
  16. 漫谈autoencoder:降噪自编码器/稀疏自编码器/栈式自编码器(含tensorflow实现)
  17. 三角形 css_CSS三角形
  18. php artisan migrate,laravel php artisan migrate错误
  19. 大数据时代网络安全分析
  20. 那些95后创业者,后来都怎样了?

热门文章

  1. FK JavaScript之:ArcGIS JavaScript API之地图动画
  2. Ubuntu12下安装redis(多图版)+ Jedis连接Redis
  3. VoIP协议标准浅析
  4. Linq to SQL学习
  5. 正则表达式的20个小应用
  6. weblogic常见漏洞
  7. iOS之深入解析如何检测“循环引用”
  8. iOS之深入解析类Class的底层原理
  9. 【数据结构与算法】之深入解析十大常用排序算法的原理分析和算法实现
  10. 1.3 Hive架构原理