【Linux系统编程】Linux第一个小程序——进度条

news2024/11/19 9:29:32

文章目录

  • 1. 对回车(\r)和换行(\n)的理解
    • 1.1 概念理解
    • 1.2 测试
  • 2. 缓冲区的理解
    • 2.1 观察现象
    • 2.2 原因解释
  • 3. 倒计时小程序
  • 4. 进度条小程序
    • 4.1 基本思路及实现
    • 4.2 改进及优化
    • 4.3 增加百分比显示
    • 4.4 增加旋转光标
    • 4.5 给进度条配色

这篇文章我们一起来完成我们Linux中的第一个小程序——进度条
在这里插入图片描述

1. 对回车(\r)和换行(\n)的理解

1.1 概念理解

在C语言中,字符可以分为可显字符(printable characters)和控制字符(control characters)。

可显字符是指可以在屏幕或打印输出上显示的字符,它们包括数字、字母、标点符号、符号等。可显字符可以直接被用户看到,并且在文本处理、显示和通信中起到重要作用。
控制字符是一类在计算机中具有特殊含义的字符,它们通常不可见或只能以特殊方式显示。这些字符用于控制文本的格式、编辑和通信等方面。

这里我们要重点理解两个控制字符——\n\r

\r表示回车,即将光标移动到当前行的起始位置
\n表示换行,即将光标向下移动一行
但是我们平时用的比如C语言打印的时候加一个\n换行
在这里插入图片描述
或者在编辑文本的时候敲enter
在这里插入图片描述
他不仅进行了换行并且光标也移到了起始位置。
但是其实这是两个步骤,先移到下一行,再移动到起始位置。
不过呢,在常见的计算机系统中,换行通常会伴随回车操作。

1.2 测试

下面我们来测试几个程序。

首先我来写一个makefile,我们待会写完代码可以直接用:
在这里插入图片描述

然后我来写一个test.c

在这里插入图片描述
那这里面我就用到了换行\n

那我来运行看一下:

在这里插入图片描述
我们看到这里就成功打印出来了hello world,并且进行了换行(且伴随了回车)。
所以后面的命令提示符就打印到了下一行,并且在开头位置。

然后我们把\n换成\r试一下:

在这里插入图片描述
再来make然后运行
在这里插入图片描述
我们看到这次什么都没打印

那为什么啥都没打印呢?

那其实就是跟最后的\r有关系,printf打印的时候,前面的hello world都没问题,但是最后遇到\r(回车),就把光标移到了最左边起始位置。
所以后面打印命令提示符的时候就把hello world覆盖掉了。

2. 缓冲区的理解

下面我们来理解一下缓冲区的概念

缓冲区(Buffer)是计算机系统中用于临时存储数据的一块内存区域。它通常用于处理输入和输出操作,以提高效率和性能。
缓冲区相当于一个中间层,位于数据的来源和目的地之间。当进行输入或输出操作时,数据先暂时存储在缓冲区中,然后再批量地传输到目标位置或从源位置读取出来。这样可以减少对源位置或目标位置的直接读写次数,从而提高数据传输效率。

2.1 观察现象

下面我们还是来观察两个程序

先看第一个:

在这里插入图片描述
这里用了一个函数sleep
sleep() 函数用于在程序中暂停执行一段时间,sleep() 函数的参数是以秒为单位的等待时间。它的作用是让程序进入休眠状态,停止执行指定的时间间隔,然后再继续执行后续的代码。
在Linux或UNIX系统中,可以包含 <unistd.h> 头文件,使用 sleep() 函数。而在Windows系统中,可以包含 <windows.h> 头文件,使用 Sleep() 函数。
然后我们观察一下结果
我这里给的是截图,这里如果大家自己测试可能会观察的更好一点
在这里插入图片描述
我们看到,这里先打印了hello world,然后进行休眠(因为我们使用了sleep)
在这里插入图片描述
休眠结束,就打印了新的命令行。

然后我们看第二个:

跟上面的区别就是我把\n去掉了
在这里插入图片描述
然后我们再来运行
在这里插入图片描述
这次我们会观察到它是先休眠
在这里插入图片描述
休眠结束然后才打印hello world,并且新的命令行直接跟在hello world后面,因为我们没有换行。

那通过对比两次程序的结果,我们能得出:

带\n的时候是先打印hello world,后休眠;而不带\n是先休眠,后打印hello world。
那这样的话,不带\n的时候,好像是先执行了sleep函数,然后才执行printf去打印。

是这样吗?

当然不是的,我们知道程序默认是按照从上到下顺序执行的。
所以肯定是先执行printf,再执行sleep,毋庸置疑。

2.2 原因解释

那为什么我们看到的是先休眠,后打印,两个程序打印的时机为什么不一样呢?

🆗,我们上面有提到缓冲区的概念:
缓冲区相当于一个中间层,位于数据的来源和目的地之间。当进行输入或输出操作时,数据先暂时存储在缓冲区中,然后再批量地传输到目标位置或从源位置读取出来。
也就是是,不管我们有没有加\n,我们的hello world这个字符串都会被暂存到缓冲区里面。
那为什么两个程序打印的时间不一样呢?
原因其实是因为两个程序的缓冲区刷新的时机不同。
在大多数编程语言和操作系统中,缓冲区被用来暂时存储要输出或被读取的数据,直到达到一定条件后才会将其发送到目标位置(如屏幕、文件、网络等)。这个条件通常是缓冲区满了、遇到换行符、或者主动进行缓冲区刷新的操作。
程序结束时,通常会自动刷新输出缓冲区。这意味着在程序执行完成后,输出缓冲区中的所有数据将被写入到相应的输出设备(如终端或控制台)并在屏幕上显示出来。
所以我们可以认为,遇到\n的时候就会触发缓冲区刷新操作。
而程序结束也会刷新缓冲区。

那现在,我想大家就明白了,为什么上面两个程序的结果有差异?

第一个程序我们加了\n,所以执行printf时遇到\n就会刷新缓冲区,那么hello world就直接显示到了显示器上。
所以是先打印,后休眠。
而第二个程序,没有\n,我们也没有手动刷新缓冲区,所以直到程序结束是刷新缓冲区,hello world 才会显示到显示器上。
因此是先休眠,后打印。

那有了缓冲区的理解,我们再来看上面最开始演示的那个程序:

在这里插入图片描述
现在在hello world后面加一个\r。
我们运行看看
在这里插入图片描述
在这里插入图片描述
休眠结束啥没打印,新的命令提示符就出来了。
那这个我们上面其实解释过,因为\r的缘故,使得光标移到了最左边起始位置,所以后面的命令提示符就把先打印出来的hello world覆盖了。
那我现在修改一下:
在这里插入图片描述
fflush这个函数可以刷新缓冲区给,那这样就相当于我们提前刷新了一下缓冲区,这样休眠就在打印后面了,方便我们观察。
此时我们再运行
在这里插入图片描述
在这里插入图片描述
这下我们就看清楚了,并不是啥也没打印。
而是hello world打印之后,光标回到了最左边,然后后面打印的命令提示符就把hello world覆盖掉了。
当然如果把\r去掉,就不会被覆盖了
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

3. 倒计时小程序

那基于上面讲的内容,我们一起来实现一个倒计时小程序练练手:

怎么做呢?

在这里插入图片描述
大家看这样写是不是就行了
这里从9开始倒计时,i从9到0,循环打印,\r保证每个数字打印之后都把光标移到起始位置,fflush刷新缓冲区,这样使得每个数字可以分开显示,每次循环i都可以刷新出来,然后休眠1秒,显示下一个数字。
我们运行看一下
在这里插入图片描述
在这里插入图片描述
这个大家可以自己写写运行一下,截图看着不方便。
具体的效果就是从9开始,9、8、7、6、5、4、3、2、1、0一次交替显示。
在这里插入图片描述
在这里插入图片描述
但是当前这样写,最终0显示完之后,这一行就被新的命令行覆盖了。
所以我们可以加一个换行
在这里插入图片描述
在这里插入图片描述
这样最后倒计时这一行就不会被覆盖了。

但是呢,我们的程序还有一些问题

我们刚才倒计时9到0都是一个数,占一个位置,所以后面的刚好覆盖前面的,那如果是从10开始呢?
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
效果就成这样了。
因为后面都是一位数,只能覆盖一个位置,后边的0就一直显示,不受影响。
实际上我们无论打印什么类型的数据,显示器上显示的内容都是一个个的字符,打印整数时,它们也会以字符的形式显示在屏幕上。计算机内部使用二进制表示整数,但在显示器上呈现给用户时,需要将其转换为对应的字符形式。

那怎么解决呢?也很简单:

我们指定域宽就行了。
printf可以用格式控制串"%md"输出域宽为m的十进制整数(默认左对齐,-m则右对齐)
在这里插入图片描述
然后我们再来运行
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
就可以了。

4. 进度条小程序

那我先来大致说一下我们最后要实现的一个进度条的样式:

就是一个大的【】,里面预留出来100个字符的空间,我们填充#,当然你也可以用其他的,1%就打印一个#,2%就两个,以此类推,后面可以显示一下具体是百分之几,随着#增加不断递增直到100%。

其实它大致的思路和上面的倒计时是一样的:

就是不断的显示并覆盖之前的内容。

那接下来我们就来实现一下。

我呢想给它写成一个多文件的形式:

在这里插入图片描述
我先创建这样3个文件。
在这里插入图片描述
先写进去这些内容。
然后把Makefile也写一下:
在这里插入图片描述

4.1 基本思路及实现

然后,我们来写实现进度条的函数process:

首先我们可以先开一个数组,把进度条需要的100个字符的空间预留出来。
那我们打印的时候可以直接打印#组成的字符串,那字符串的话就要再给\0开一个空间。
在这里插入图片描述
然后我们可以给buf数组全部初始化为\0,这样我们后续添加#就不用考虑\0的问题了。
在这里插入图片描述
然后我们循环打印并不断添加#就行了,当然我们这里还应该使用\r不断的回车,使每一次新打印的覆盖之前的,并且每次循环printf之后要使用fflush刷新缓冲区,这样才能每次循环都够打印出来内容,要不然程序结束之前一直留存在缓冲区。
在这里插入图片描述
我们运行看看效果:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

4.2 改进及优化

上面的实现,根据实际的运行效果我们可以发现两个问题:

首先第一个休眠时间设置成1秒有点长了,这样跑到100%需要100秒

所以我们可以选择把sleep函数换成usleep。
sleep的参数是以秒为单位的,而usleep是以微秒为单位的。
我们可以设置成0.1秒休眠时间
在这里插入图片描述
运行一下:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
这次速度确实快了

但是
第二个问题:进度条这一行显示完毕,新出现的命令行会把进度条的一部分覆盖掉。

怎么解决?
很简答,加个换行就行了
在这里插入图片描述
再看:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
这次就没事了。

当然我可以加一个宏,这样后面替换进度条的样式就很方便:

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

然后我们再修改一下,改成这种:

在这里插入图片描述
类似一个箭头,改一下代码:
在这里插入图片描述
看一下效果
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
但是这样最后停下来,还有一个箭头,好像有点不好看。
向前推进的时候显示箭头,100%的时候不显示,我们再来修改一下
加一个判读就行了
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

4.3 增加百分比显示

那一般进度条后面还有显示百分比,我们也来加一个:

在这里插入图片描述
运行一下
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

4.4 增加旋转光标

然后我们再来在后面增加一个旋转光标:

可以通过循环显示这四个字符| / - \来模拟一个旋转的过程(注意\要用转义字符\\
在这里插入图片描述
我们来运行看看效果
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
就可以了。

4.5 给进度条配色

在C语言中,可以使用ANSI转义序列来输出不同的颜色。ANSI转义序列是一系列的字符组合,用于控制终端的文本样式和颜色。

关于配色方案,网上可以找到很多相关的资料,大家有兴趣可以按照自己的喜好去配色,我这里简单演示一下:

在这里插入图片描述
比如我配个红色
在这里插入图片描述
在这里插入图片描述
来个绿色
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

还有背景颜色也可以设置:

在这里插入图片描述
试一下,来个青色
在这里插入图片描述

也可以同时设置字体颜色/前景颜色和背景颜色

在这里插入图片描述
我们来个字体红色,背景青色
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

大家可以按自己的喜好设置。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/756172.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于FT232HL的USB2.0转ARINC429板卡

基于FT232HL的USB2.0转ARINC429板卡 1 概述 《USB2.0转ARINC429板卡》采用底板子板&#xff0c;层叠安装的结构&#xff1b;使用同样的底板&#xff0c;变换不同功能的子板实现不同的功能版本。 a) 降低硬件设计复杂度&#xff1a;新板卡设计只需要设计子板&#xff0c;子板的…

类和对象(C++)( static成员、explicit、友元、内部类、匿名对象)

类和对象 static成员概念static成员“登场”特性static成员使用 注意 explicit从一段代码引入explicit和explicit相关特性 友元友元函数引入问题解决 小结 友元类 内部类概念特性 匿名对象引入使用 static成员 概念 类的静态成员&#xff1a;声明为static的类成员。 静态成员变…

【黑客】网络安全靠自学?只会毁了你!

1️⃣网安现状 ❗本文面向所有 想要涉足网安领域 或 已经涉足但仍处在迷茫期 的伙伴&#xff0c;如果你月薪达到了3w&#xff0c;那么请你离开。 如果没有&#xff0c;希望你继续看下去&#xff0c;因为你人生的转折点将从这篇文章开始。 ✈️网络安全&#xff0c;一个近几年大…

5 个能出色完成数据恢复的免费数据恢复软件知识分享。

有时&#xff0c;由于意外删除或某些问题&#xff0c;您可能会丢失 Windows 10 笔记本电脑或台式机上的重要数据。Windows 操作系统不提供任何内部工具来恢复已删除的数据。但是有一些非常好的数据恢复软件可以更专业地完成这项工作。最好的人总是有报酬的&#xff0c;但不用担…

按键精灵、auto.js等一些移动端脚本 如何连接云服务器的数据库, 进行读写操作

一、技术背景 按键手机版和auto.js&#xff0c;只支持连接本地数据库sqllite&#xff0c;该数据库只存在本地 其他设备无法读写&#xff0c;就像本地的txt一样。 而很多脚本作者的需求是&#xff1a;多个脚本&#xff0c;甚至在全国不同城市的脚本也能读取和写入同一批数据&…

AJAX-day01

(创作不易&#xff0c;感谢有你&#xff0c;你的支持&#xff0c;就是我前行的最大动力&#xff0c;如果看完对你有帮助&#xff0c;请留下您的足迹&#xff09; 目录 AJAX 概念和 axios 使用 什么是 AJAX 怎么用 AJAX &#xff1f; axios 使用 认识 URL 什么是 URL&…

韦东山Linux教学视频中的makefile文件详细介绍

前言 &#xff08;1&#xff09;在学习韦东山Linux教学视频的时候&#xff0c;他的makefile并没有做详细的介绍。以至于我学了很长时间对他的makefile文件不理解。所以本文将会详细介绍韦东山Linux教学视频中的makefile文件含义。 &#xff08;2&#xff09;注意&#xff1a;我…

使用 docker-compose 部署 Jenkins

注&#xff1a;我是在虚拟机&#xff08;Ubuntu&#xff09;上部署了 docker-compose&#xff0c;然后才使用 docker-compose 部署 Jenkins&#xff01; 关于如何在 Ubuntu 部署 docker-compose&#xff0c;可以看我其它的文章。 本文目录 1. 创建 docker_jenkins_compose 目录…

【NI USRP】每一个USRP是如何命名的呢,和原厂Ettus型号有什么关联呢?

详细的硬件配置&#xff0c;非常有助于设备的选型。 如果您采购了X310子板&#xff0c;是可以将其转化为对应的USRP型号的设备。 B系列 EttusNI-USRP频段最大带宽通道B200mini无70 MHZ - 6 GHZ56 MHz1X1B200mini-i无70 MHZ - 6 GHZ56 MHz1X1B205mini-i无70 MHZ - 6 GHZ56 MHz…

三菱以太网通讯模块在哪

捷米特JM-ETH-FX采用工业级设计&#xff0c;导轨安装&#xff0c;带通讯线。不占用PLC编程口&#xff0c;上位机通过以太网对PLC数据监控的同时&#xff0c;触摸屏可以通过复用接口X2与PLC进行通讯。捷米特JM-ETH-FX支持工控领域内绝大多数SCADA软件&#xff0c;支持三菱MC以太…

C#开发的OpenRA游戏之维修按钮

C#开发的OpenRA游戏之维修按钮 前面分析物品的变卖按钮,如果理解这个流程,再看其它按钮的流程,其实是一样的,所以前面的文章是关键,只有理解通透的基础之上,才能继续往下。 维修按钮的存在价值,就是当建筑物受到敌方破坏,还没有完全倒掉之前,可以使用金币来进行修理。…

java项目之电子商城系统(ssm+mysql+jsp)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于ssm的电子商城系统。技术交流和部署相关看文章末尾&#xff01; 开发环境&#xff1a; 后端&#xff1a; 开发语言&#xff1a;Java 框架&…

对抗业务逻辑攻击:传统安全工具为何失效

随着数字环境的不断发展&#xff0c;不良行为者寻求利用应用程序漏洞的策略也在不断发展。最阴险的攻击类型之一是业务逻辑攻击 (BLA)。与可以通过签名或模式识别的已知攻击&#xff08;例如 SQL 注入攻击&#xff09;不同&#xff0c;BLA 针对应用程序内的核心功能和决策过程。…

python_股票增加控制人与流通股东等筛选条件

目录 写字前面&#xff1a; 结果展示 获取数据 行业数据 控制人数据 十大流通股东数据 开始合并 1 从行业数据中提取证券股的行业数据 2 合并控制人数据 3 合并十大流通股东 4 把三个结果按列合并 写字前面&#xff1a; 在分析数据的时候&#xff0c;常常需要的字段…

Hippo4j监控RabbitMQ框架的线程池

&#x1f680; 线程池管理工具-Hippo4j &#x1f680; &#x1f332; AI工具、AI绘图、AI专栏 &#x1f340; &#x1f332; 如果你想学到最前沿、最火爆的技术&#xff0c;赶快加入吧✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;CSDN-Java领域优质创作者&#…

【分布式系统管理框架】Zookeeper集群

分布式系统管理框架 1. Zookeeper1.1 Zookeeper概述1.2 Zookeeper工作机制1.3 Zookeeper特点1.4 Zookeeper数据结构1.5 Zookeeper应用场景1.6 Zookeeper选举机制 2.部署Zookeeper集群3. 知识点总结3.1 zookeeper3.2 zookeeper选举机制 1. Zookeeper 1.1 Zookeeper概述 Zookee…

什么是微服务架构

什么是微服务架构&#x1f349; 你考虑过吗&#xff1f;什么是微服务&#xff0c;为什么越来越多的企业&#xff0c;为了使自己构建的应用满足客户的期望&#xff0c;而和微服务架构进行整合呢&#xff1f; 微服务&#xff0c;又叫微服务架构&#xff0c;是一种软件架构方式。…

web-报错注入

必要的函数 rand select rand(0) from hackbiao; rand(0)&#xff1a;生成以0开头的随机数&#xff0c;生成的数量与字段下数据的条数相等。如果i没有这个地段的话&#xff0c;就会自己形成一个新的字段打印出来。 count和group by grouip by在进行排序的时候&#xff0c;会…

网络线程模型

堵塞IO模型&#xff1a;每个连接都由独立的线程进行处理。当并发度较高时系统资源占用较大&#xff0c;并且如果线程发生了IO堵塞还会浪费线程资源Reactor模型&#xff1a;reactor线程监听&#xff0c;并分发事件给相应的handlerProactor模型&#xff1a;交由系统进行异步处理&…