【Linux】进度条:行缓冲区、\r 与 fflush 的实战

编写一个命令行进度条程序,看起来是个小练习,但背后涉及行缓冲区回车与换行的区别 ,以及fflush的使用。这些概念在更深层次的 I/O 编程中会反复出现。

1. 回车与换行

先厘清两个历史概念:

  • 换行(Line Feed,\n:光标下移一行。

  • 回车(Carriage Return,\r:光标回到行首。

老式打字机时代,换行和回车是两个独立动作------打字到行尾后,先"咔嗒"推杆让滚筒转一行(换行),再手动把压字架推回左端(回车)。到了计算机终端,二者被合并为"按下回车键执行换行+回车"。但 \n\r 在 C 语言中仍然保持各自独立的本意。

2. 行缓冲区

看三段代码:

代码 1

c

复制代码
// 打印"hello\n",等3秒结束
printf("hello\n");
sleep(3);

现象:立刻显示"hello",然后等3秒。\n 触发了行缓冲刷新。

代码 2

c

复制代码
// 打印"hello",等3秒结束
printf("hello");
sleep(3);

现象:3 秒内屏幕无任何输出,3 秒后程序结束才显示"hello"。因为输出被暂存在行缓冲区里,等到程序结束或缓冲区满才刷新。

代码 3

c

复制代码
// 打印"hello",手动刷新
printf("hello");
fflush(stdout);
sleep(3);

现象:立刻显示"hello"。fflush(stdout) 强制刷新标准输出缓冲区。

3. 倒计时程序

理解了行缓冲和 \r,就可以写一个简易倒计时:

c

复制代码
int i = 10;
while (i >= 0) {
    printf("%2d\r", i);   // \r 回到行首,覆盖之前的数字
    fflush(stdout);        // 立刻输出
    i--;
    sleep(1);
}
printf("\n");

\r 把光标拉回行首,下一次 printf 会覆盖同一行上的旧内容,形成就地倒计时的视觉。

4. 进度条代码

把倒计时的思路扩展成进度条,可以做到在调用方处理下载或计算任务时实时刷新进度:

c

复制代码
void FlushProcess(double total, double current) {
    char buffer[101] = {0};
    static const char* label = "|/-\\";
    static int cnt = 0;
    int len = strlen(label);

    int num = (int)(current * 100 / total);
    for (int i = 0; i < num; i++)
        buffer[i] = '=';

    double rate = current / total;
    cnt %= len;
    printf("[%-100s][%.1f%%][%c]\r", buffer, rate * 100, label[cnt]);
    cnt++;
    fflush(stdout);
}

%-100s 表示左对齐占据100个字符宽度,不足处用空格填充。旋转字符 |/-\\ 通过循环索引给出动画效果。每次调用 FlushProcess 刷新同一行,形成连续推进的视觉效果。

usleep(50000) 即 50 毫秒的延迟,让动画不至于闪得太快。实际项目中用不到 usleep,这里只是为了人眼能看清过程。

相关推荐
掘金者阿豪9 分钟前
写了很多内容后,我还是决定给自己搭一个Typecho博客
后端
Younglina30 分钟前
打了3年羽毛球球才发现:我对自己的装备和胜率一无所知
前端·后端
Go_error1 小时前
Datatypes:Go 轻松支持数据库JSON类型
后端·go
长大19881 小时前
新手必踩 Redis 10 个低级坑:过期时间、KEYS 命令、持久化误区
后端
Csvn2 小时前
Python 两大经典坑点 —— 可变默认参数 & 闭包延迟绑定
后端·python
Csvn2 小时前
定时任务 — Crontab 从入门到生产实战
后端
ServBay3 小时前
Laravel Herd MCP 的替代,多语言与跨平台的 AI 本地开发选择
后端·ai编程·mcp
GoGeekBaird4 小时前
Prompt、Context、Harness 工程全景图
后端
SimonKing4 小时前
艹,维护AI写的代码,我心态崩了......
java·后端·程序员