0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

探索C语言入门基础之缓冲区

Android编程精选 来源:编程学习总站 作者:写代码的牛顿 2021-06-28 17:24 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

01

C标准库缓冲区探索

在计算机里缓存是一个很重要的概念,C标准库里大量使用了缓存,最为典型的就是标准输入和标准输出的缓存,关于C语言的输入和输出看这篇文章即可,利用好缓存可以大幅提高程序性能,首先我们看一下下面这段代码会输出什么?

#include 《stdio.h》 #include 《unistd.h》 int main() { printf(“Hello World!”); //往标准输出输出字符串 //程序停留在while循环里,程序退出会强制刷新缓冲区数据 while(1){ sleep(1); } return 0; }

我们在程序里调用printf函数打算在标准输出“Hello World!”,下面的while(1)循环是想让程序停在这里不退出程序,每次睡眠1s避免占用大量CPU资源,在Linux中包含unistd.h头文件才能使用sleep函数。现在我们编译以下看看会输出什么?

ce807804-d751-11eb-9e57-12bb97331649.png

我们看到,其实什么都没有输出。但是从程序上看,我们已经调用了printf函数往标准输出输出字符串,这就是缓存在起作用了。printf函数默认是行缓冲,当输出字符串里有 或者行缓冲区被填满或者手动调用fflush函数才会一次性将数据输出。现在你只要加上一条语句输出换行符,就能在标准输出输出字符串了。

printf(“ ”); //换行,默认标准输出会立即输出刷新缓冲区

或者我们手动调用fflush也可以强制刷新缓冲区,输出字符串。

fflush(stdout); //强制刷新标准输出缓冲区

往标准错误输出字符串的语句编译运行后会发生什么呢?

fprintf(stderr, “error information”); //往标准错误输出输出信息

fprintf函数将信息往第一个FILE指针类型参数输出,这里第一个参数我们传入stderr,编译运行后立即在控制台上输出字符串“error information”。标准错误输出和标准输出运行测试结果对比我们知道,调用fprintf函数往标准错误输出信息时不需要加字符‘ ’,也不需要强制刷新缓冲区也能立即输出信息。这是因为标准错误输出是无缓冲模式,写入什么数据就立即输出什么数据。

下面我们再看看输入代码

#include 《stdio.h》 int main() { char arr[100] = {0}; scanf(“%s”, arr); return 0; }

在这段代码里,程序运行后我们从标准输入输入数据,直到按下回车才将数据输入到数组arr里。在按下回车后,实际上刷新了输入缓冲区将数据一次性写入到数组arr里。

03

缓冲区的作用

在计算机里应用程序调用一个系统调用从用户态进去内核态再将结果回到用户态开销较大。如果我们调用printf函数,每次输出一个字符都要从用户态切换到内核态,那么连续输出多个字符开销成本将会非常大,这个时候缓存就起到非常大的作用了,输出的字符串先在应用程序里缓存起来,缓存到一定数量后再调用系统调用一次性将缓存数据输出到标准输出。

由于只调用了一次系统调用,比连续调用多个系统调用性能高上不少。在生活中我们也能感受到缓存带来的效率提升,打个比方你办公室有一个垃圾桶,楼下有倒垃圾的地点,如果扔一个垃圾到垃圾桶里我们就拿去倒掉,将会在办公室和楼下之间来回很多趟,浪费大量时间。如果将垃圾桶装满,再一次性拿到楼下倒掉,只需要跑一次就能把垃圾全都倒掉,节省了时间,提高了效率。

04

缓冲模式和使用方式

C语言里有行缓冲模式、全缓冲模式和无缓冲模式。

行缓冲模式:填满缓冲区或者有换行符‘ ’或者调用fflush函数强制刷新缓冲区会立即输出。

全缓冲模式:填满缓冲区或者调用fflush函数强制刷新缓冲区会立即输出。

无缓冲模式:写入什么数据就会立即输出什么数据,例如标准错误输出默认的缓冲模式。

下面我们用实际代码演示如何使用三种缓冲模式,设置缓冲模式会用到setvbuf函数,我们先来看看setvbuf函数声明。

/* Make STREAM use buffering mode MODE. If BUF is not NULL, use N bytes of it for buffering; else allocate an internal buffer N bytes long. */ extern int setvbuf (FILE *__restrict __stream, char *__restrict __buf, int __modes, size_t __n) __THROW;

第一个参数是FILE类型指针,第二个参数是外部缓冲区指针,第三个参数是缓冲模式,第四个参数是缓冲大小,如果不使用外部缓冲区,函数内部将会调用malloc申请一块内存作为内部缓冲区。

形参mode提供了三个参数分别设置不同的缓冲区模式

_IONBF unbuffered _IOLBF line buffered _IOFBF fully buffered

无缓冲模式实例代码

#include 《stdio.h》 #include 《unistd.h》 int main() { setvbuf(stdout, NULL, _IONBF, 0); //标准输出设置为无缓冲,不使用外部缓冲区 printf(“Hello World!”); return 0; }

编译运行会立即输出

Hello World!

行缓冲模式实例代码

#include 《stdio.h》 #include 《unistd.h》 int main() { setvbuf(stdout, NULL, _IOLBF, 0); //标准输出设置为行缓冲模式,不使用外部缓冲区 printf(“how are you”); //不会立即输出字符串 fflush(stdout); //强制刷新缓冲区,立即输出字符串 return 0; }

编译运行后,由于调用了fflush会强制刷新数据到标准输出。

全缓冲模式实例代码

#include 《stdio.h》 #include 《unistd.h》 int main() { setvbuf(stdout, NULL, _IOFBF, 0); //标准输出设置为全缓冲模式,不使用外部缓冲区 printf(“Hello World!”); //不会立即输出 printf(“how are you”); //不会立即输出 printf(“ ”); while(1){ sleep(1); } return 0; }

编译运行后发现没有任何输出,现在我们在while循环前面加上下面这条语句,编译运行看看。

fflush(stdout); //强制刷新缓冲区

编译运行后立即输出了字符串!

同样的使用方式可以用于标准输入和标准错误输出,只需要把stdout缓存stdin或者stderr即可。

编辑:jq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 数据
    +关注

    关注

    8

    文章

    7361

    浏览量

    95128
  • 计算机
    +关注

    关注

    19

    文章

    7848

    浏览量

    93557
  • C语言
    +关注

    关注

    183

    文章

    7649

    浏览量

    146342
  • 函数
    +关注

    关注

    3

    文章

    4423

    浏览量

    67994
  • 代码
    +关注

    关注

    30

    文章

    4981

    浏览量

    74506

原文标题:C语言入门基础之缓冲区

文章出处:【微信号:AndroidPush,微信公众号:Android编程精选】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    DMA传输完成通知未在S32K324上的半满缓冲区时触发,为什么?

    S32K344发生第一次中断时(半缓冲区已满并发生中断): 第一次中断发生时S32K324: 我使用以下代码片段来配置 DMA 传输: K344 上的 TCD: TCD 对S32K324:
    发表于 04-02 08:10

    C语言缓冲区(缓存)详解

    缓冲区又称为缓存,它是内存空间的一部分。也就是说,在内存空间中预留了一定的存储空间,这些存储空间用来缓冲输入或输出的数据,这部分预留的空间就叫做缓冲区。   缓冲区根据其对应的是输入设
    发表于 01-14 07:30

    CW32L052串口的缓冲区机制

    默认缓冲区配置 CW32L052的UART模块支持硬件FIFO(通常为16字节),但HAL库或用户代码需手动管理接收缓冲区。若未显式分配足够大的软件缓冲区,可能导致数据溢出。 HAL库缓冲区
    发表于 11-24 06:40

    飞凌嵌入式ElfBoard-文件I/O的了解探究I/O缓冲

    \");}return 0;}编译运行并查看测试结果:112345123451234512345^C设置缓冲区长度为9字节、全缓冲,首次打印1并刷新缓冲区,此时按回车等待后面打印,循环
    发表于 11-19 16:24

    飞凌嵌入式ElfBoard-标准IO接口设置缓冲区

    1.setvbuf 用于以对文件的 stdio 缓冲区进行设置,譬如缓冲区缓冲模式、缓冲区的大小、起 始地址等。 1)头文件 #include 2)函数原型 int setvbuf
    发表于 11-14 09:02

    【道生物联TKB-623评估板试用】+3、模块深度测评:高频率数据传输的缓冲区陷阱与优化方案

    模块深度测评:高频率数据传输的缓冲区陷阱与优化方案 本文记录了在实际项目中使用620 LoRa模块时遇到的一个隐蔽却关键的问题,以及完整的排查和解决方案。 问题现象:高频率发送数据时的异常行为
    发表于 10-15 19:29

    移植的lvgl,在运行的时候,缓冲区无法释放怎么解决?

    代码在运行的时候,只有lvgl线程用于刷新,另一个线程只有一个串口打印。 当运行一段时间后,发现,程序会在LVGl中,lv_refr.c这个库下面第625行代码, 在这一直判断,看介绍说是在等待释放缓冲区,求大神给个思路
    发表于 09-09 07:28

    USB如何判断IN缓冲区有值?

    调用 USBDevice->BulkInEndPt->XferData 可获取 USB IN 缓冲区的数据大小(按字节),但数据会被传至主机。目前,我先要判断 USB
    发表于 07-23 08:21

    对于CYUSB3014芯片,哪个函数可以读取GPIF II端DMA缓冲区中缓存的实际数据量?

    : 1.哪个函数可以读取USB端口缓冲区数据的实际字节数? 2. 哪个函数可以读取 GPIF II 上 DMA 缓冲区数据的实际字节数? 3、GPIF II 接口的标志信号会受 USB 端口状态的影响吗?会怎样,然后又会产生怎样的影响?
    发表于 07-21 07:27

    如何清除CYUSB3014的缓冲区数据?USB接口数据什么时候发送到电脑?

    、如何清除GPIF II接口处对应的DMA BUFFER数据?当标志设置为满/非满状态时,一旦 DMA 缓冲区已满,标志信号就会指示它已满。清除DMA缓冲区数据后,相应的标志信号会改变吗?会不会变成非满状态
    发表于 07-18 07:58

    USB缓冲区中的内容满了之后,是否有标志位进行反馈?

    USB缓冲区中的内容满了之后,是否有标志位进行反馈。
    发表于 07-17 07:13

    请问USB缓冲区取数据可以多次取吗?

    在使用USB软件获取数据是,下位机给我发送了13个32位数据到USB IN缓冲区,为什么我调用API函数想要第一次取1个32位数据,取完之后再取12位数据,程序会卡死。
    发表于 07-16 08:12

    socket缓冲区溢出的原因?怎么解决?

    我在测试视频通话时 发现丢帧特别严重 进行了一些列的排查 发现socket本身似乎有问题 通过测试代码发现了大量的缓冲区溢出我尝试换了不同的服务器 我还分别测试了wifi网卡和4G网卡 全都这样
    发表于 06-19 06:34

    【RA4L1-SENSOR】串口收发 + 环形缓冲区

    (); #endif } 环形缓冲区代码 #define MAX_BUFFER_LEN100 typedef enum { QUEUE_ERR, QUEUE_SUCCESS }QUEUE; typedef
    发表于 06-11 10:24

    解析RZ/N2L CANFD模块的缓冲区机制(3)

    CANFD模块的缓冲区机制,帮助工程师更高效地管理CAN消息,提高系统性能。(下面的内容主要涉及RZN2L CANFD外设手册的解读,篇幅较长,感兴趣的读者可以收藏,以备日后不时之需)
    的头像 发表于 05-20 13:53 1486次阅读
    解析RZ/N2L CANFD模块的<b class='flag-5'>缓冲区</b>机制(3)