初识Linux —— 第一个程序(进度条)

初识Linux —— 第一个程序(进度条)

前言

已经学习了linux下基本工具的使用,现在来实践练习一下。

1.回车和换行

Windows下,我们认为回车换行是一个概念;但事实上,换行就是换到下一行的当前位置,而回车是回到当前行的开头位置。

我们之所以会认为回车和换行是一个概念,那是因为在我们使用\n的时候,它做了回车和换行两个操作。

现在来看linux下这样两段代码

#include<stdio.h>intmain(){printf("迟来的grown\n");return0;}
#include<stdio.h>intmain(){printf("迟来的grown\r");return0;}
在这里插入图片描述

可以看到\n\r的不同,但运行结果就不一样,其中\r就是表示回车;

那为什么\r回车就没有显示出来结果呢?

这里就要了解缓冲区这个东西了。

缓冲区又称为缓存,它是内存空间的一部分。也就是说,在内存空间中预留了一定的存储空间,这些存储空间用来缓冲输入或输出的数据,这部分预留的空间就叫做缓冲区。

我们可以大致理解为在输入输出时,并不是一个一个字符来进行的(部分特殊除外),这里是行缓冲区。

行缓冲:当输入和输出时,遇到换行符,才执行真正的I/O操作。这时我们输入的字符就会先存放在缓冲区,当按下回车键才进行实际的I/O操作。典型的就是stdinstdout

行缓冲,输出时要遇到换行符才执行真正的输出;所以\r回车,输出的内容就存放在了缓冲区,在程序结束时才清空缓冲区。

这里看两段代码
#include<stdio.h>intmain(){printf("迟来的grown\n");sleep(3);return0;}
intmain(){printf("迟来的grown");sleep(3);return0;}

这里运行结果:(看到的)

第一段代码先输出了迟来的grown,然后再休眠了3秒第二段代码休眠了3秒,在程序结束时才输出了迟来的grown

这两段代码的差别就是\n,这也证明行缓冲区,遇到换行刷新缓冲区的内容。(在程序结束时,也会刷新缓冲区内容)

看到这里可能有疑问,既然程序结束后也会刷新缓冲区的内容,那为什么使用\r在程序结束后也没有输出结果啊?

这是因为linux中它要输出命令行信息,在程序执行完之后,光标是在行开头的,命令行信息就覆盖了要输出的内容。

那我们能不能进行一些操作来看到要输出的内容呢?

当然是有的,我们可以使用fflush来刷新缓冲区(stdout)的内容。

intmain(){printf("迟来的grown");fflush(stdout);sleep(3);return0;}

以上代码运行结果

输出了迟来的grown,再休眠了3秒,最后程序结束;命令行信息将输出内容覆盖了。

2.倒计时程序

了解了回车换行,现在我们来简单些一个倒计时程序。
#include<stdio.h>intmain(){int count =10;while(count){printf("%-2d\r",count);fflush(stdout); count--;sleep(1);}}

代码如上,这里可能出现的一些问题

输出时只覆盖了第一个数字:使用%2d/%-2d,我们要让数字占两个数字(字符)的位置。程序运行不显示倒计时:使用fflush刷新缓冲区内容。

3.进度条程序

在进行编写第一个linux程序——进度条之前,我们先重温一下makefile

BIN=progress SRC=$(shell ls *.c) OBJ=$(SRC:.c=.o) CC=gcc RM= rm -rf $(BIN):$(OBJ) $(CC) -o $@ $^ %.o:%.c $(CC) -c $< .PHONY: clean: $(RM) $(OBJ) $(BIN) 
首先shell ls *.c,获得当前目录下所有的.c文件,可以使用wildcard *.cSRC:.c=.o,将SRC中所以的.c后缀改为.o$@$^$@是获取依赖文件列表,$^获取目标文件。$<,是将依赖文件列表拿出来逐个执行。

更加详细的make/makefile深入了解Linux —— make和makefile自动化构建工具_linux makeself-ZEEKLOG博客

好现在,我们正式开始写linux下的第一个程序——进度条

先来看一下我们要实现的进度条是什么样的:

进度条程序

在这里插入图片描述

我们想要看到的进度条是,进度一直在增加,百分比一直在上涨,最后的方框内一直在转圈。

那该如何实现呢?

首先先来看一下需要写哪些文件

在这里插入图片描述
这里呢,写了四个文件makefile文件code.c:程序主函数main所在的文件progress.c:进度条程序的代码源文件progress.h:进度条程序的代码头文件

了解了这些,现在来看我们如何实现呢?

这里虽然我们看起来是进度条连续增加的,但是事实上就是一次一次输出的结果。

不知道你是否还记得在实现贪吃蛇小游戏的时候,我们就是一次次输出,来实现蛇的移动;这里也同理,我们依然一次次打印来达到我们预期的效果。

某一时刻进度条的输出

我们先来看某一时刻进度条是如何输出的;

假设现在我们正在下载一个软件,软件大小1024.00MB ,我们当前下载了512.00MB,那如何打印这一时刻的进度条呢?

voidprocess(){double total =1024.00;//下载总量double current =512.00;//当前下载量int rate =(int)(current*100)/total;//进度条百分比,也是要打印`=`的数量constchar* str ="|-/\\";//表示最后方框内旋转的字符,最后一个\\转义字符staticint cnt =0;//表示当前应该那个字符,来转圈char buff[101]={'\0'};int i=0;for(i=0;i<rate;i++){ buff[i]='=';}printf("[%-100s][%d%%][%c]\n",buff,rate,rate[cnt]);}
在这里插入图片描述

可以看到在一时刻的进度条就已经成型了;那我们根据先已经写好的代码,来实现完整的进度条。

进度条实现

我们知道了这一时刻的进度条如何打印,那我们之间让current当前进度累加即可(这就涉及到一个速度问题)。还有一点,最后方框内如何让它转动起来

为了控制速度,我们既可以控制current累加的值,当然也可以控制休眠时间usleep

为了控制方框内转动,我们定义一个静态常量,来决定应该输出哪一个字符。

具体代码如下

#include"progress.h"#include<unistd.h>#include<string.h>#defineNUM101#defineCH'='voidprocess(){char buff[NUM];memset(buff,0,sizeof(buff));constchar* str="|/-\\";int len =strlen(str);int cnt =0;while(cnt<=100){printf("[%-100s][%d%%][%c]\r",buff,cnt,str[cnt%len]);fflush(stdout); buff[cnt]=CH;usleep(10000); cnt++;}printf("\n");}

这里curremt增加量和usleep休眠时间可以自行修改。

对于这个版本,感觉还是差点意思如果我们需要下载内容,大小不一,下载网速也不同,那该如何?

这个进度条我们只有修改这个函数内的数值才能控制进度条的快慢,我们还可以进行修改,通过传参来控制总量。

这里提供修改后的版本,细节就不详细讲解了。

当然这个版本也存在一些不足之处,有待提高。

code.c

#include"progress.h"#definespeed1.0voidDownload(double total){double current=0;while(current<=total){Process(total,current);usleep(6000); current+=speed;}printf("\ndownload %lfMB Done\n",total);}intmain(){//process();Download(1024.00);return0;}

progress.h

#include<stdio.h>#include<unistd.h>voidProcess(double total,double current);

progress.c

voidProcess(double total,double current){char buff[NUM];memset(buff,0,sizeof(buff));constchar* str ="|/-\\";int len =strlen(str);//当前进度int num =(int)(current*100)/total;int i=0;for(i=0;i<num;i++){ buff[i]=CH;}staticint cnt =0; cnt%=len;printf("[%-100s][%d%%][%c]\r",buff,num,str[cnt]); cnt++;fflush(stdout);}

*到这里本篇内容就结束了,希望对你有所帮助。

制作不易,感谢大佬的支持。

我的博客即将同步至腾讯云开发者社区,邀请大家一同入驻:https://cloud.tencent.com/developer/support-plan?invite_code=2oul0hvapjsws

Read more

Flutter for OpenHarmony:tostore 鸿蒙原生 KV 数据库,支持 SQL 与 NoSQL 混合存储(全能型数据引擎) 深度解析与鸿蒙适配指南

Flutter for OpenHarmony:tostore 鸿蒙原生 KV 数据库,支持 SQL 与 NoSQL 混合存储(全能型数据引擎) 深度解析与鸿蒙适配指南

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net 前言 在移动应用开发中,数据持久化(Data Persistence)永远是架构设计中不可或缺的一环。无论是保存用户的登录状态、偏好设置,还是缓存新闻列表、聊天记录,选择一个合适的数据库往往决定了 App 的运行流畅度和开发效率。 在 Flutter 生态中,我们熟知的数据库方案琳琅满目: * Shared Preferences: 轻量级,但只适合存简单的 Key-Value,性能较差,且不支持复杂查询。 * Sqflite: 基于 SQLite 的封装,功能强大且稳定,但它是关系型数据库,Schema 变更(数库迁移)极其繁琐,且需要编写大量的 SQL 语句或依赖复杂的 ORM。 * Hive: 纯 Dart 编写的 NoSQL 数据库,速度极快(

By Ne0inhk
【MySQL数据库基础】(三)MySQL 库的核心操作全解析:创建、修改、备份一条龙搞定

【MySQL数据库基础】(三)MySQL 库的核心操作全解析:创建、修改、备份一条龙搞定

前言         在 MySQL 的学习和实战中,数据库(库)的操作是最基础也是最核心的环节,无论是项目开发、数据管理还是运维维护,都绕不开库的创建、配置、修改、备份等一系列操作。很多刚接触 MySQL 的小伙伴容易在字符集、校验规则、备份恢复这些细节上踩坑,今天这篇文章就结合实战案例,把 MySQL 库的全套操作讲透,从基础语法到高级技巧,从避坑指南到实战演示,让你一文掌握 MySQL 库操作的精髓! 一、创建数据库:基础语法与个性化配置         创建数据库是操作 MySQL 的第一步,看似简单的一句命令,背后却藏着字符集、校验规则的关键配置,选对配置能让后续的开发和数据管理少走很多弯路。 1. 核心创建语法         MySQL 中创建数据库的官方语法如下,其中大写部分为关键字,中括号[]内的为可选项,也是实际开发中需要重点关注的部分: CREATE DATABASE [IF NOT EXISTS]

By Ne0inhk
Flutter 组件 chopper_built_value 适配鸿蒙 HarmonyOS 实战:强类型网络层架构,构建不可变模型与高性能序列化闭环

Flutter 组件 chopper_built_value 适配鸿蒙 HarmonyOS 实战:强类型网络层架构,构建不可变模型与高性能序列化闭环

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 组件 chopper_built_value 适配鸿蒙 HarmonyOS 实战:强类型网络层架构,构建不可变模型与高性能序列化闭环 前言 在鸿蒙(OpenHarmony)生态迈向大规模企业级应用、涉及高频网络数据交互、复杂业务模型及严苛运行时稳定性的背景下,如何确保网络请求返回的数据在进入 UI 层前具备绝对的类型安全,已成为衡量应用架构“护城河”深度的核心标准。在鸿蒙设备这类强调 AOT 极致性能与低容错率的环境下,如果应用依然依赖动态类型的 Map<String, dynamic> 进行数据传递,由于由于后端字段变更或类型溢出,极易由于由于运行时强转失败导致应用在关键业务路径上的红屏崩溃。 我们需要一种能够实现自动化代码生成、支持不可变(Immutable)模型且具备拦截器解耦能力的序列化粘合层。 chopper_built_value 为 Flutter 开发者引入了将 Chopper

By Ne0inhk
基于 Rust 与 DeepSeek 大模型的智能 API Mock 生成器构建实录:从环境搭建到架构解析

基于 Rust 与 DeepSeek 大模型的智能 API Mock 生成器构建实录:从环境搭建到架构解析

前言 在现代软件工程中,API 接口的开发与前端联调往往存在时间差。为了解耦前后端开发进度,Mock 数据(模拟数据)的生成显得尤为关键。传统的 Mock 数据生成依赖于静态 JSON 文件或简单的规则引擎,难以覆盖复杂的业务逻辑与语义关联。随着大语言模型(LLM)的兴起,利用 AI 根据 Schema 定义动态生成高保真的模拟数据成为可能。本文详细记录了使用 Rust 语言结合 DeepSeek-V3.2 模型构建智能 Mock 生成器的完整技术路径,涵盖操作系统层面的环境准备、Rust 工具链的深度配置、代码层面的异步架构设计以及编译期的版本兼容性处理。 第一部分:Linux 系统底层的构建环境初始化 Rust 语言的编译与链接过程高度依赖于底层的系统工具链。Rust 编译器 rustc 在生成二进制文件时,需要调用链接器(Linker)将编译后的对象文件(Object Files)与系统库(

By Ne0inhk