搜索
查看: 1010|回复: 0

[分享] linux的 “行缓冲”机制

[复制链接]

该用户从未签到

2586

主题

2613

帖子

0

蝴蝶豆

版主

最后登录
2021-3-16
发表于 2021-3-1 15:21:50 | 显示全部楼层 |阅读模式
1. 前言
“我的人品有这么差吗!半个小时Linux都没调度我的线程!”。
前些日子同事抱怨,“linux是多线程系统,每个线程都有一个时间片,为什么我的程序似乎一直没被调度”,随即给我演示执行过程。大致归纳成下面的代码逻辑。
程序启动后仅仅输出一行“run”,然后就没有然后了,喝着咖啡、嗑着瓜子、带薪摸鱼,屏幕上干干净净。理想状态下每10秒执行任务完毕后输出“do”。
分析代码逻辑可知 “推断” 代码是否被运行的依据是屏幕上输出 “do” ,”而这真的科学吗?答案是否定的,linux有着雨露均沾的调度系统,很难做到半小时没被调度。问题就出在linux的 “行缓冲”机制

  1. void main()
  2. {
  3.     printf("run\n");
  4.     while(1) {
  5.         sleep(10);
  6.         do_thing();
  7.         printf("do");
  8.     }
  9. }
复制代码


2. 行缓冲

上文的printf输出目标设备由STDOUT指定,STDOUT可能指向115200波特率的串口设备,也可能指向本地图形设备,相对于CPU而言他们的速度要慢上几个数量级。都是为了提高机器或者程序的性能、提高CPU利用率。协调高速设备和低速设备之间速度的不匹配,操作系统默认在标准I/O上采用行缓冲机制,printf的内容首先存储在内存,当缓冲填满或检测到换行符’’时则输出到目标设备
Linux上这个行缓冲默认是1024Byte,每10s输出do占用2Byte,为了等待缓冲填满需要等待81min。
知道原理后解决方法就简单了。
方法1:fflush强制刷新标准输出stdout

  1. #include
  2. void main()
  3. {
  4.     printf("run\n");
  5.     while(1) {
  6.         sleep(10);
  7.         do_thing();
  8.         fflush(stdout);
  9.         printf("do");
  10.     }
  11. }
复制代码

方法2:输出“换行符”

  1. #include
  2. void main()
  3. {
  4.     printf("run\n");
  5.     while(1) {
  6.         sleep(10);
  7.         do_thing();
  8.         printf("do\n");
  9.     }
  10. }
复制代码

回复

使用道具 举报

您需要登录后才可以回帖 注册/登录

本版积分规则

关闭

站长推荐上一条 /3 下一条

Archiver|手机版|小黑屋|论坛-意法半导体STM32/STM8技术社区

GMT+8, 2024-4-29 18:13 , Processed in 1.199061 second(s), 29 queries .

Powered by Discuz! X3.4

Copyright © 2001-2024, Tencent Cloud.

快速回复 返回顶部 返回列表