Linux第一个小程序 之 【进度条】
目录
1.回车与换行的概念
2.行缓冲区
3.进度条小程序
3.1makefile
3.2processBar.h
3.3processBar.c
3.4main.c
4.倒计时
1.回车与换行的概念
在C语言中:
\r(回车):光标移动到当前行的开头
\n(换行):光标移动到下一行(不同系统实现不同)
实际效果取决于操作系统:Windows需要\r\n才能正确换行,而Linux/Unix只需要\n
2.行缓冲区
#include <stdio.h>int main(){printf("hello Makefile!\n");sleep(3);return 0;}
上述代码的结果是先打印字符串hello Makefile!,停3秒,再换行出现命令行
#include <stdio.h>int main(){printf("hello Makefile!");sleep(3);return 0;}
上述代码的结果是先停3秒,再打印出字符串hello Makefile!,后面紧跟命令行
这是因为\n会刷新缓冲区,将字符串打印出来并换行
不加任何格式控制,字符串将在缓冲区中保存,当程序结束时,缓冲区才会被刷新,字符串才会被打印出来,由于未换行,字符串后面紧跟命令行
#include <stdio.h>int main(){printf("hello Makefile!");fflush(stdout);sleep(3);return 0;}
上述代码的结果是先打印字符串hello Makefile!,停3秒,再紧跟着出现命令行
fflush函数强制刷新了缓冲区
3.进度条小程序
3.1makefile
目标文件:依赖文件
依赖方法
伪目标
processbar:processBar.c main.cgcc -o $@ $^.PHONY:clean
clean:rm -f processbar
头文件 processBar.h 与源代码在同一目录时:
使用 #include "processBar.h" 编译器能自动找到
编译指令中不需要用 -I. 指定当前目录(但可以用)
不应该直接编译头文件,应该编译包含它的源文件
直接编译头文件可能导致"未定义引用"错误,因为只有声明没有实现
3.2processBar.h
#pragma once#include <stdio.h>#define NUM 101
#define BODY '='
#define Top 100
#define RIGHT '>'extern void processbar(int rate);
extern void initBar();
编译指令#pragma once,防止头文件被重复包含
包含头文件<stdio.h>,定义一些宏
关于extern关键字:
变量:在头文件中声明全局变量时必须使用extern,否则可能引发重复定义错误
函数:函数声明默认具有external链接属性,extern关键字可以省略
作用:告诉编译器该标识符在其他翻译单元中定义,链接时再解析
存储分配:extern声明不会分配存储空间,只是引用性声明
3.3processBar.c
#include"processBar.h"
#include<unistd.h>//usleep的头文件
#include<string.h>const char* label = "|/-\\";
//字符数组,存放进度条
//字符数组声明为全局时,自动初始化全为'\0'
char bar[NUM];void processbar(int rate)
{if(rate < 0 || rate > 100) return;memset(bar, BODY, rate);if(rate <= 99) bar[rate] = RIGHT;int len = strlen(label);//进度条中显示100个字符,左对齐,双%转义为%,//回车,每次从头开始显示printf("[%-100s][%d%%][%c]\r", bar, rate, label[rate%len]);//\r并不会刷新缓冲区,\n会//不刷新导致sleep结束之后才会显示字符fflush(stdout);if(rate == 100) return;//进度为100不用休眠了//微秒时间的休息,1s = 10^6us usleep(50000);}void initBar()
{memset(bar, '\0', sizeof(bar));
}
默认进度条由100个字符组成,再加上\0,字符数组一共有101个元素
processBar函数根据参数rate,打印当前进度条的长度
关于打印:%-100s表示左对齐,宽度为100的字符串,%d%%,打印当前进度百分比,%c,模拟进度扩展时的旋转圈,\r回车,每次从头开始打印,因为rate持续增大,所以效果就是覆盖写,并逐渐增加
initBar函数就是重置字符数组
3.4main.c
#include "processBar.h"
#include<unistd.h>typedef void (*callbak_t)(int);//函数指针类型void downLoad(callbak_t cb)
{int total = 1000, cur = 0;while(cur <= total){//下载....usleep(50000);//模拟下载时间int rate = cur*100/total;//下载进度cb(rate);//显示进度cur += 10;//下载了一部分}printf("\n");
}int main()
{initBar();printf("downLoad1:\n"); downLoad(processbar);initBar();printf("downLoad2:\n"); downLoad(processbar);initBar();printf("downLoad3:\n"); downLoad(processbar);initBar();printf("downLoad4:\n"); downLoad(processbar);return 0;
}
downLoad函数回调函数cb进而完成进度的显示
每次现在前需要清空进度条

4.倒计时
#include<stdio.h>int main()
{int cnt = 10;while(cnt >= 0){printf("%-2d\r", cnt);//刷新缓冲区,不然等休眠之后才会显示fflush(stdout);sleep(1);--cnt;}printf("\n");return 0;
}
- \r 使光标回到行首
确保每个数字都在同一行的同一位置显示
避免了数字竖向排列或累积显示
- %-2d 表示左对齐,至少占2个字符宽度
确保两位数和一位数都占用相同的显示空间
避免了 10 → 9 时显示 90 的问题(因为9无法完全覆盖10)
- 标准输出通常是行缓冲的(遇到\n才刷新)
强制立即输出缓冲区内容
确保每个数字都能实时显示,而不是等到程序结束
