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

fgets()(函数原型:char *fgets(char *restrict str, int size, FILE *restrict stream))

这个函数原型不太好看出个所以然来,可以理解为(char *fgets(“容器的地址”, “容器的大小”, “从哪里读取”))


一般用法:

char a[100] = {0};

fgets(a, 100, stdin);

通俗来讲的话,fgets()函数的作用就是用来读取一行数据的。但要详细且专业的说的话,fgets()函数的作用可以这么解释:从第三个参数指定的流中读取最多第二个参数大小的字符到第一个参数指定的容器地址中。在这个过程中,在还没读取够第二个参数指定大小的字符前,读取到换行符'\n'或者需要读取的流中已经没有数据了。则提前结束,并把已经读取到的字符存储进第一个参数指定的容器地址中。

在正常情况下fgets()函数的返回值和它第一个参数相同。即读取到数据后存储的容器地址。但是如果读取出错或读取文件时文件为空,则返回一个空指针。


fgets()函数的运行流程大概是这样子的:

当系统调用这个函数的时,系统便会阻塞等待用户的输入,直到用户输入回车符’\n’才返回程序。然后用户输入的内容会被系统放进输入缓存区里面,fgets()函数便会进来读取其“第二个参数减1(为什么减1后面说)”个字节存进它第一个参数指向的内存地址中,如果在还没读取够需要的字节大小前读取到换行符’\n’则提前返回。


fgets()函数的注意事项1

fgets()函数的最大读取大小是其“第二个参数减1”,这是由于字符串是以’\0’为结束符的,fgets()为了保证输入内容的字符串格式,当输入的数据大小超过了第二个参数指定的大小的时候,fgets()会仅仅读取前面的“第二个参数减1”个字符,而预留1个字符的空间来存储字符串结束符’\0’

测试代码:

#include <stdio.h>
int main(void)
{
char a[10] = {0};
printf("你的输入:");
fgets(a, 4, stdin);
//printf("%s\n", a);//下面这句的输出和这句是一样的
printf("printf(\"%%s\\n\", a)%c==>%s\n", ';', a);
return 0;
}

运行效果:

在这个例子中,fgets()的第二个参数是4,所以它最多只能存储输入的(4-1 = 3)个字符进第一个参数指向的地址空间里面。输入“abcde”,数组a[]中只有“abc”。


fgets()函数的注意事项2

fgets()函数的眼里,换行符’\n’也是它要读取的一个普通字符而已。在读取键盘输入的时候会把最后输入的回车符也存进数组里面,即会把’\n’也存进数组里面,而又由于字符串本身会是以’\0’结尾的。所以在输入字符个数没有超过第二个参数指定大小之前,你输入n个字符按下回车输入,fgets()存储进第一个参数指定内存地址的是n+2个字节。最后面会多出一个’\n’和一个’\0’,而且’\n’是在’\0’的前面一个(\n\0)。

测试代码:

#include <stdio.h>
int main(void)
{
char a[10] = {0};
printf("你的输入:");
fgets(a, 10, stdin);
//printf("%s\n", a);//下面这句的输出和这句是一样的
printf("printf(\"%%s\\n\", a)%c==>%s\n", ';', a);
for(int i=0; i<10; i++)
{
if(a[i] == '\n')
printf("a[%d]是换行符'\\n'\n", i);
if(a[i] == '\0')
printf("a[%d]是字符串结束符'\\0'\n", i);
}
return 0;
}

运行效果:



在这个例子中,由于输入的字符小于参数2指定的最大读取字符数,所以fgets()函数会把换行符’\n’也储存进数组a[]里面,在运行界面的第三行哪里换了两次行,就是由于这个多出来的换行符’\n’和我输出代码中的换行符’\n’共同作用的结果。


fgets()函数的注意事项3

fgets()函数只负责读取,并不会事先清空参数1指向的地址内存。读取到的字节会覆盖原地址储存,但没有覆盖到的内容还是保持原样。

测试代码:

#include <stdio.h>
int main(void)
{
char a[10] = {'1','1','1','1','1','1','1','1','1','1'};
printf("你的输入:");
fgets(a, 10, stdin);
//printf("%s\n", a);//下面这句的输出和这句是一样的
printf("printf(\"%%s\\n\", a)%c==>%s\n", ';', a);
for(int i=0; i<10; i++)
{
if(a[i] == '\n' || a[i] == '\0')
printf("a[%d] = '\\%c'", i, a[i]=='\n'?'n':'0');
else
printf("a[%d] = %c", i, a[i]);
printf("\n");
}
return 0;
}

运行结果:


fgets()函数的注意事项4

在用fgets()函数读取键盘输入的时候,如果输入多于其“第二个参数减1”个字符大小的数据,fgets()只会读取走前”第二个参数减1”个字符,多余的字符残留在输入缓存区里面。如果不清空,可能会影响下次输入。

测试代码:

#include <stdio.h>
int main(void)
{
char a[4] = {0};
char b[10] = {0};
printf("存进a的输入:");
fgets(a, 4, stdin);
for(int i=0; i<4; i++)
printf("a[%d] = %c\n", i, a[i]);
printf("存进b的输入:");
fgets(b, 10, stdin);
printf("这里没有阻塞等待输入,而是直接跳过了\n");
//printf("%s", a);//下面这句的输出和这句是一样的
printf("printf(\"%%s\", b)%c==>%s", ';', b);
return 0;
}

运行结果:

在这个例子中,输入“abcde”之后,数组a[]读取走“abc”之后,代码运行到第11行的时候并没有停下来等待用户的输入,而是直接读取了还留在缓存区里面的“de\n”,读取到‘\n’之后返回,所以我最后一行的输出代码中并没有加上换行符’\n’,因为数组b[]中已经包含有换行符’\n’了。


fgets()函数的注意事项5

遇到再更新。。。


原博客始发于CSDN,在如今博客界的转载抄袭泛滥的环境下,原创不易,点个赞再走呗。以下是博客首页的链接。


零BUG是原则性问题。

fgets()函数的详解以及使用时需要注意的一些细节-C语言基础的更多相关文章

  1. C语言对文件的操作函数用法详解1

    在ANSIC中,对文件的操作分为两种方式,即: 流式文件操作 I/O文件操作 一.流式文件操作 这种方式的文件操作有一个重要的结构FILE,FILE在stdio.h中定义如下: typedef str ...

  2. Go语言Slice作为函数参数详解

    Go语言Slice作为函数参数详解 前言 首先要明确Go语言中实质只有值传递,引用传递和指针传递是相对于参数类型来说. 个人认为上诉的结论不对,把引用类型看做对指针的封装,一般封装为结构体,结构体是值 ...

  3. Scala进阶之路-Scala函数篇详解

    Scala进阶之路-Scala函数篇详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.传值调用和传名调用 /* @author :yinzhengjie Blog:http: ...

  4. php中的PDO函数库详解

    PHP中的PDO函数库详解 PDO是一个“数据库访问抽象层”,作用是统一各种数据库的访问接口,与mysql和mysqli的函数库相比,PDO让跨数据库的使用更具有亲和力:与ADODB和MDB2相比,P ...

  5. C语言对文件的操作函数用法详解2

    fopen(打开文件) 相关函数 open,fclose 表头文件 #include<stdio.h> 定义函数 FILE * fopen(const char * path,const  ...

  6. MySQL UUID函数的详解(转)

    MySQL UUID函数的详解 MySQL中可以有二类用于生成唯一值性质的工具:UUID()函数和自增序列,那么二者有何区别呢?我们就此对比下各自的特性及异同点: l  都可以实现生成唯一值的功能: ...

  7. 自写函数VB6 STUFF函数 和 VB.net 2010 STUFF函数 详解

    '*************************************************************************'**模 块 名:自写函数VB6 STUFF函数 和 ...

  8. SQL Server数据库ROW_NUMBER()函数使用详解

    SQL Server数据库ROW_NUMBER()函数使用详解 摘自:http://database.51cto.com/art/201108/283399.htm SQL Server数据库ROW_ ...

  9. PHP函数篇详解十进制、二进制、八进制和十六进制转换函数说明

    PHP函数篇详解十进制.二进制.八进制和十六进制转换函数说明 作者: 字体:[增加 减小] 类型:转载   中文字符编码研究系列第一期,PHP函数篇详解十进制.二进制.八进制和十六进制互相转换函数说明 ...

  10. PHP date函数参数详解

    PHP date函数参数详解 作者: 字体:[增加 减小] 类型:转载       time()在PHP中是得到一个数字,这个数字表示从1970-01-01到现在共走了多少秒,很奇怪吧 不过这样方便计 ...

随机推荐

  1. Spark log4j 配置

    Spark的ml包提供了非常好用的调参功能,通过ParamGridBuilder构建待选参数(如:logistic regression的regParam),然后数据量小的时候可以用CrossVali ...

  2. vs找不到msvcp120d .dll,无法继续执行代码。重新安装可能会解决此问题。

    原文链接:https://blog.csdn.net/qq_24537165/article/details/90137317 环境:win10 vs2015 c++ opencv3.4.0 截图: ...

  3. sprinboot多个子模块下 依赖包没有找到 解决方案

    最近因为在使用springboot开发项目,在开发过程中,发现自己的子模块导入通用的模块 在启动中 说找不到这个类 百度下 说我要在pom文件下 pulus 插件 那里 加上这段代码 <conf ...

  4. mathcurve.com

    https://mathcurve.com/surfaces.gb/surfaces.shtml

  5. ubuntu20.04系统openjdk11变更openjdk-8-jdk

    ubuntu20.04系统openjdk11变更openjdk-8-jdk一.卸载openjdk11先检查是否安装,命令:dpkg --list | grep -i jdk移除openjdk包,命令: ...

  6. 90、java ftp 读取文件

    https://blog.csdn.net/qq_38380025/article/details/80679128

  7. 【剑指Offer】【链表】合并两个排序的链表

    题目:输入两个单调递增的链表,输出两个链表合成后的链表,当然我们需要合成后的链表满足单调不减规则. A:若链表1为空,则合并后的链表头结点为pHead2:若链表2为空,则合并后的链表头结点为pHead ...

  8. Linux下的zip和tar压缩解压缩命令详解

    一.zip压缩工具 zip的压缩包在windows和linux中都比较常用,它可以压缩目录和文件,压缩时录时,需要指定目录下的文件.zip后面先跟目标文件名,即压缩后得自定义压缩包名,然后跟要压缩的文 ...

  9. etcd使用Cfssl生成自签证书(pem)

    CFSSL是CloudFlare开源的一款PKI/TLS工具,CFSSL包含一个命令行工具和一个用于签名,验证并且捆绑TLS证书的HTTP API服务,环境构建方面需要 Go 1.12+. 需要两套证 ...

  10. iOS 15 UI适配

    1. UINavigationBar 在iOS 15中,UINavigationBar默认为透明.在滑动时会有模糊效果.如果想要一直就是模糊效果,可以通过改变scrollEdgeAppearance属 ...