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

这篇文章要探讨的是“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

遇到再更新。。。


零BUG是原则性问题。

  • 103
    点赞
  • 229
    收藏
    觉得还不错? 一键收藏
  • 9
    评论
在 C 语言中,`fgets` 函数用于从文件中读取一行字符串。它的函数原型如下: ```c char *fgets(char *str, int n, FILE *stream); ``` `fgets` 函数接受三个参数: - `str`:指向一个字符数组的指针,用于存储读取的字符串。 - `n`:指定要读取的最大字符数(包括换行符和空字符),以避免缓冲区溢出。 - `stream`:指向要读取的文件的指针。 `fgets` 函数会从文件中读取一行字符,并将其存储在 `str` 指向的字符数组中。它会读取包括换行符在内的所有字符,然后在末尾添加一个空字符。 以下是一个示例,展示了如何使用 `fgets` 函数从文件中逐行读取内容: ```c #include <stdio.h> int main() { FILE *file = fopen("data.txt", "r"); // 打开文件以只读模式 if (file == NULL) { printf("无法打开文件!\n"); return 1; } char line[100]; // 逐行读取文件内容 while (fgets(line, sizeof(line), file) != NULL) { printf("%s", line); } fclose(file); // 关闭文件 return 0; } ``` 上述代码中,我们首先使用 `fopen` 函数打开一个名为 "data.txt" 的文件,指定打开模式为 "r",即只读模式。 然后,我们定义了一个字符数组 `line`,用于存储每行读取的内容。 接下来,我们使用一个 `while` 循环和 `fgets` 函数来逐行读取文件的内容。当 `fgets` 返回值不为 `NULL` 时,表示成功读取了一行内容,我们将其打印出来。 最后,我们使用 `fclose` 函数关闭文件。 请注意,`fgets` 函数会将换行符也一同读取,并且会在字符串末尾添加一个空字符。因此,打印输出时需要使用 `%s` 格式化字符,而不是 `%c`。
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值