虽然用 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。

下面写一个程序:

#include <stdio.h>
int main(void)
{char str[20]; printf("请输入一个字符串:");fgets(str, 7, stdin); printf("%s\n", str);return 0;
}

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

下面写一个程序看一下:

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

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

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

 include <stdio.h>
int main(void)
{char str[30];char ch;printf("请输入字符串:");fgets(str, 29, stdin);printf("%s", str);  //后面不要加'\n'scanf("%c", &ch);printf("ch = %c\n", ch);return 0;
}

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

【转载】fgets函数用法详解相关推荐

  1. ROW_NUMBER() OVER()函数用法详解 (分组排序 例子多)

    ROW_NUMBER() OVER()函数用法详解 (分组排序 例子多) https://blog.csdn.net/qq_25221835/article/details/82762416 post ...

  2. mysql: union / union all / 自定义函数用法详解

    mysql: union / union all http://www.cnblogs.com/wangyayun/p/6133540.html mysql:自定义函数用法详解 http://www. ...

  3. C++ search()函数用法详解(深入了解,一文学会)

    find_end() 函数用于在序列 A 中查找序列 B 最后一次出现的位置.那么,如果想知道序列 B 在序列 A 中第一次出现的位置,该如何实现呢?可以借助 search() 函数. search( ...

  4. C++ reverse()函数用法详解(深入了解,一文学会)

    reverse_copy() 算法可以将源序列复制到目的序列中,目的序列中的元素是逆序的.定义源序列的前两个迭代器参数必须是双向迭代器.目的序列由第三个参数指定,它是目的序列的开始迭代器,也是一个输出 ...

  5. fgets()函数的详解-使用技巧-C语言基础

    这篇文章要探讨的是"fgets()函数的详解以及使用时需要注意的一些细节".涉及fgets()函数的应用和需要注意的问题.属于C语言基础篇(持续更新). fgets()(函数原型: ...

  6. EXCEL公式VLOOKUP函数用法详解

    EXCEL公式VLOOKUP函数用法详解 示例下载 VLOOKUP函数 在表格或数值数组的首列查找指定的数值,并由此返回表格或数组中该数值所在行中指定列处的数值. 这里所说的"数组" ...

  7. python lambda函数-Python的Lambda函数用法详解

    在Python中有两种函数,一种是def定义的函数,另一种是lambda函数,也就是大家常说的匿名函数.今天我就和大家聊聊lambda函数,在Python编程中,大家习惯将其称为表达式. 1.为什么要 ...

  8. C++ copy()函数用法详解(深入了解,一文学会)

    C++ 算法 copy() 函数用于将容器 [first,last] 的所有元素从结果开始复制到不同的容器中. 本文介绍了copy.strcpy.strncpy.memcpy.copy_n.copy_ ...

  9. fgets()函数的详解以及使用时需要注意的一些细节-C语言基础

    这篇文章要探讨的是"fgets()函数的详解以及使用时需要注意的一些细节".涉及fgets()函数的应用和需要注意的问题.属于C语言基础篇(持续更新). fgets()(函数原型: ...

最新文章

  1. iOS三种录制视频方式详细对比
  2. AI一分钟 | 小米公布Q2财报,上市以来股价振幅高达30%;俄制造商推出步行杀手机器人...
  3. C# 对象深拷贝、浅铐贝、直接拷贝(转)
  4. excel二极管伏安特性曲线_【刘敏蔷老师】半导体二极管的原理及应用
  5. 增量式pid调节方式有何优点_增量式pid和位置式pid相比各有什么优缺点
  6. 织梦dedecms财务会计代理记账财税公司网站模板 带手机版
  7. tomcat相关实验
  8. oracle设置缓冲区大小设置,描述Oracle优化库高速缓冲区
  9. Python爬取猫眼电影排行榜并写入MySQL
  10. java 铁路管理信息系统_java多线程之铁路售票系统
  11. 深入浅出Python——Python高级语法之函数
  12. 问答WAP版重新改版上线
  13. CAD快捷键命令 mac版
  14. MySQL中的窗口函数
  15. 国土档案管理信息系统【档案著录】-地籍类档案著录
  16. win7安装JDK并配置
  17. ubuntu1604 grep sed 正则表达式
  18. 74HC595移位寄存器的使用
  19. 微信将可开小号!微信内测一个手机可注册俩号
  20. APP自动化测试框架搭建(五)--Python+Appium+pytest-html

热门文章

  1. Qt:C++应用程序开发入门
  2. 2020年Q3房产行业网络关注度分析报告
  3. python namedtuple
  4. linux下tomcat查看端口
  5. vps系统服务器,vps系统和云服务器
  6. 学计算机投影仪定义,一种计算机教学用投影仪射头的制作方法
  7. Oracle 利用储存过程插入循环插入大量数据方法
  8. VSCode配置MPX开发
  9. python数字分割
  10. OAI——单虚拟机多basic核心网部署