👦个人主页:Weraphael
✍🏻作者简介:目前正在学习c++
和Linux
还有算法
✈️专栏:Linux
🐋 希望大家多多支持,咱一起进步!😁
如果文章有啥瑕疵,希望大佬指点一二
如果文章对你有帮助的话
欢迎 评论💬 点赞👍🏻 收藏 📂 加关注😍
一、回车和换行
回车和换行是两个不同的概念。
- 换行: 通常用符号
\n
表示。作用是将光标移动到下一行的开头,即在文本中换行。
- 回车指的是:将光标移动到当前行的开头位。用
\r
表示。
printf
打印的时候,hello
其实是正常打印的,但是遇到了\r
(回车),就将光标移动到当前行的开头位。于是Linux
就把hello
给覆盖掉了
二、缓冲区
为了更好展示缓冲区的现象,在此之前先认识一个Linux
下的函数sleep
。
sleep()
函数是一个系统调用,用于在C
语言程序中暂停执行一段时间。(可以通过man
手册来了解)
其中,seconds
参数表示暂停的时间长度(单位为秒)。在函数调用期间,程序会进入睡眠状态,不会进行任何操作,直到指定的时间过去后恢复执行。
接下来言归正传,来看看以下代码
这肯定难不倒大家。结果一定是hello Linux
,并且在程序结束前会延迟2
秒
那如果我将以上代码稍作修改(仅仅是将'\n'
去掉了)
有的人想,这还需要问吗,结果肯定和上面一样!可是结果真的是这样吗?
好像和我们想的有点不太一样,以上的结果是先执行sleep
函数停顿2
秒,再执行printf
输出的hello Linux
。
可是执行顺序真的是以上这样的吗?
当然不是!学过C语言的都清楚,代码是自上而下运行,无论上面哪种情况,理应先输出字符,再休眠。那么字符串hello Linux
去哪了?虽然不知道在哪?但能保证它一定是存在的。那么它存哪了呢?
答:缓冲区。缓冲区是指一块内存区域,用于临时存储数据,以便在需要时进行读取或写入。缓冲区通常用于提高程序的效率,减少对底层资源(如磁盘、网络等)的频繁访问次数。
- 在
C
语言中,printf
函数是自带缓冲区的,这意味着它不会立即将输出内容打印到终端,而是先将内容存储在缓冲区中,然后在适当的时候再将内容刷新到终端上显示。因此,即使printf
函数执行了,你可能不会立即看到输出结果。而加入sleep
函数之后,程序会暂停执行2
秒钟,这段时间足够让缓冲区中的内容被刷新到终端上(程序终止也会强制刷新缓冲区)。- 如果输出的内容包含换行符
\n
,printf
函数会将其视为刷新缓冲区的信号。因此,如果你的printf
语句以\n
结尾,结果就会立即显示出来,而不需要显式调用sleep
函数或等待缓冲区满。
如果我不想用\n
就能立马刷新呢?有何办法?
可以使用
fflush
函数来手动刷新缓冲区。当你调用fflush(stdout)
时,它会强制将标准输出流(stdout
)的缓冲区中的内容立即刷新到输出设备(如终端)上显示,而不管缓冲区是否已满或者是否遇到了换行符。
接下来,看看加上fflush
函数的效果
上面说过遇到换行\n
时,输出缓冲区会被自动刷新。那么遇到回车\r
,输出缓冲区会被自动刷新吗?我们可以一起做个实验
如上结果所示:回车\r
不会触发刷新缓冲区,需要使用fflush
函数来手动刷新缓冲区。
三、实现倒计时程序
首先以9
秒举例:
以上的代码逻辑比较简单,我们直接运行程序看看效果
为了保证代码的正确性,我们以10
秒开始倒计时来看看效果
我们发现效果有点瑕疵。
实际上,无论打印什么类型的数据,显示在显示器上的内容都是一个个字符。显示器是由一个个像素组成的。每个像素可以显示不同的颜色或者字符。当你打印不同类型的数据时,它们会被转换为字符形式并逐个显示在显示器上。
解决的办法也很简单,指定域宽就行。
然后来看看效果
效果比之前好多了,那左对齐是否就更好看些呢?
来看看最终效果
四、实现进度条
4.1 准备工作
为了提高代码的可维护性、可扩展性和代码复用性,我们采取多文件方式
为了方便编译代码,再把makefile
文件写出来,其内容如下:
我们先把大致的框架写出来
4.2 实现步骤之基础版
需求:进度条在一行不断向右移动,到达
100%
后停下来
我们可开一个长度为101
的字符数组,其中100
个存储进度条样式,可以是=
、-
等等(根据需求来定),最后多出来的字符我们可以给'\0'
。那么起始我们可以将字符数组全部初始化为'\0'
进度条可以参考倒计时。进度为0%
的时候不打印进度字符串,进度为1%
,打印一个进度字符,以此类推。那如何保证在一行可以让进度条不断向右移动?我们可以使用回车\r
,并且每打印完一次进度字符,后面的打印个数总要比前一次多1
。
我们可以看看效果
我们发现,以上打印速度太慢了,循环100
次,那么就要打印100
秒。因此这里再介绍一个函数usleep
,它和sleep
函数功能一样,唯一不同的是单位不一样,它是以微妙为单位的。
那么如果循环100
,我们可以规定10s
内打印完,那么循环一次就要0.1
秒,转化为微妙就是1e5
要将
0.1
秒转换为微秒,可以使用以下公式:
1s = 1000毫秒
1毫秒 = 1000微秒
- 因此,
微秒 = 秒 × 1000000
根据以上公式,将
0.1
秒转换为微秒的计算如下:
0.1秒 = 0.1 × 1000000 = 100000 微秒
我们再来看看效果
确实比之前快很多!好像还是有点瑕疵,右]
好像随着精度条一起移动,而我们希望的是想让其固定下来。那么我们可以为此指定域宽,预留出100
个空间给进度字符
我们再来看看效果
还是有点瑕疵,怎么是从右往左,我们希望进度条是从左向右。
这是因为:在C
语言中,格式化输出默认是右对齐的,可以通过在指定宽域数字前加一个-
。。
再来看看效果
4.3 实现步骤之增加百分比
增加百分比就非常简单了,直接看代码吧
需要注意的是:在许多编程语言中,包括Java、C、C++
等,百分号%
被用作格式化字符串的占位符,用来表示将要被替换的部分。因此,如果你想要在输出中打印一个百分号符号,需要使用两个百分号来进行转义。
我们来看看效果
4.4 实现步骤之增加箭头
对边界判断即可,判断条件为i == 100
时,就不应该出现箭头>
,那么当i < 100
时,箭头可以一直存在
来看看程序效果
4.4 实现步骤之增加旋转光标
我们可以类似于缓存的旋转光标。字符为| / - \
。
需要注意的是:在C
语言中,反斜杠\
是一种转义字符。因此如果想要打印\
,需要将其转义\\
来看看效果吧
五、源代码
Gitee
仓库链接:点击跳转