s_gets()函数解析

        在学习《C primer plus》时,有一个获取输入的函数s_gets()出现频率非常高,此处记录下对这个函数的理解,该函数首次出现于程序清单11.10。

函数源代码:

char *s_gets(char *st, int n)  
{  
    char *ret_val;  
    char *find;  
    ret_val = fgets(st, n, stdin);  
    if(ret_val)  
    {  
        find = strchr(st, '\n');  
        if(find)  
        {  
            *find = '\0';  
        }  
        else  
        {  
            while(getchar() != '\n')  
                continue;  
        }  
    }  
    return ret_val;  
}  

测试代码:

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>

int main(int argc, char *argv[])
{
    char test[10] = {0};
    while(1)
    {
        s_gets(test, 10);
        printf("output: %s \n", test);
    }

    return 0;
}

函数解析:

(1)fgets()

/**
 * @brief fgets()函数获取一整行输入:
 *          读取输入直达换行符(换行符也被读取),或者读到文件结尾,或者读取(n-1)个字符
 *          读取完成后,再在读取内容末尾添加空字符'\0',使读取内容成为一个字符串
 * @param str 指向一个字符数组,该数组存储了要读取的字符串。
 * @param n 这是要读取的最大字符数
 * @param stream FILE*指针,指明要读入的文件,如果是键盘输入,则以stdin(标准输入)作为参数
 * @return char* 如果返回成功,则返回值与char* str参数相同,如果到达文件末尾或者没有读取到任何字符,str 的内容保持不变,并返回一个空指针
 *               如果发生错误,返回一个空指针。
 */
char *fgets(char *str, int n, FILE *stream)

        可以发现fgets()的限制条件可以分为两类,字符串长度(包括回车'\n')大于(n-1),则读取(n-1)的长度,并在末尾加上空字符'\0',如果字符串长度(包括回车'\n')小于(n-1),则表示fgets获取了“一整行”,将字符串(包括回车'\n')读取至char* str中。

(2)strchr()

/**
 * @brief strchr()函数寻找参数 str 所指向的字符串中搜索第一次出现字符 c(一个无符号字符)的位置
 * 
 * @param str 
 * @param c 
 * @return char* 该函数返回在字符串 str 中第一次出现字符 c 的位置,如果未找到该字符则返回 NULL
 */
char *strchr(const char *str, int c)

        获取输入字符串st后,寻找st中是否有 '\n' 字符,如果有,则替换为 '\0' 字符,使st成为一个字符串。这实际上对应的情况就是键盘输入的字符串长度(包括'\n')小于或者等于(n-1)的情况,这样 '\n' 字符才会存在于st中。

(3)

        else中对应的情况是键盘输入的字符串长度大于(n-1),那么st存储了(n-1)个字符后,再加上1个 '\0' ,成为字符串。此时在输入输出缓冲区中,还存在着没被st存储的剩余的字符,如果不处理这些字符,那么在主程序的下一次读取中,会从缓冲区读取这些剩余字符。

        else
        {
            while(getchar() != '\n')
                continue;
        }

        做个实验,如果将s_gets()函数中的的else{}部分注释掉,输入超过10个字符,比如1234567890abcdefg。那么结果如下,可以发现输出了两次:

        注释掉else{}部分的话,会发现输出了两次,取消注释,再次编译运行的话,只会输出一次。在while()循环中,getchar()每次从输入缓冲区取出一个字符,直至取到 '\n' 换行符,此时缓冲区的剩余字符就全部取出了(包括 '\n') ,这样就不会影响下一次s_gets()获取键盘输入了。

(4)补充

        输入输出缓冲区是在内存中开辟的一块空间,键盘输入是发送到输入缓冲区,程序再从缓冲区取数据的。缓冲区在代码中是没有具体体现的,需要脑内有缓冲区这个模型。

为什么需要缓冲区:

详解getchar()函数与缓冲区_学无止境-CSDN博客_getchar缓冲区

举个例子,我们知道计算机CPU的处理速度很快的,而我们键盘的输入速度总是比不过CPU的处理速度,那么CPU就得一直等着键盘输入完,这样很浪费资源。于是,我们党键盘输入完了,再让CPU一次性处理,这样就会大大地提高效率。

又比如,我们的打印机打印文档,打印机的处理速度是很慢的,所以我们会将文档输出到打印机的缓存中去,这样打印机就可以自行慢慢打印,而不必占用CPU资源。

有时候程序中使用了sleep(),此时使用printf()等输出函数可能会没有即时输出,要等程序运行结束后才输出,有可能是输出缓冲区的问题:

printf()函数缓存区刷新问题_h___q的博客-CSDN博客_printf刷新输出

解决方法有两种,一种是强制刷新流,另一种是设置输出缓冲区大小为0:

fflush(stdout); //强制刷新流
setvbuf(stdout,NULL,_IONBF,0); //设置输出缓冲区为0

  • 17
    点赞
  • 40
    收藏
    觉得还不错? 一键收藏
  • 5
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值