嵌入式——Linux系统的使用以及编程练习
目录
一、Linux的进程、线程概念
(一)命令控制进程
1、命令查看各进程的编号pid
2、命令终止一个进程pid
二、初识Linux系统的虚拟机内存管理
(一)虚拟机内存管理
(二)与STM32内存管理对比
三、Linux调用函数编程
具体操作:
四、总结
一、Linux的进程、线程概念
Linux中的进程(Process)和线程(Thread)是操作系统进行任务调度的核心概念。
- 进程是资源分配的基本单位,每个进程拥有独立的内存空间、文件描述符等系统资源。进程之间相互隔离,需通过进程间通信(IPC)交换数据。
- 线程是进程内的执行单元,属于同一进程的多个线程共享进程的内存和资源(如代码段、全局变量)。线程独立调度,切换开销小,适合需要并发执行的场景。
在Linux中,线程通过轻量级进程(Lightweight Process, LWP)实现,底层由clone()系统调用创建。内核使用task_struct结构统一管理进程和线程,区别在于资源共享程度:线程属于同一线程组(TGID相同),共享内存;而不同进程的TGID不同,资源独立。
(一)命令控制进程
如表为部分控制进程二点命令,但是此处我们着重操作查看进程编号和终止进程
| 分类 | 命令 | 功能描述 |
|---|---|---|
| 查看进程信息 | ps | 列出当前进程快照 |
| top / htop | 动态监控进程资源占用 | |
| 终止进程 | kill | 通过PID发送信息终止进程 |
| killall | 通过进程名批量终止 | |
| 调整优先级 | nice | 启动新进程时设置优先级(范围:-20~19) |
| renice | 修改已运行进程的优先级 | |
| 后台进程管理 | nohup | 忽略挂断信号,使进程在终端关闭后仍运行 |
| jobs/fg/bg | 管理当前终端的后台作业 |
1、命令查看各进程的编号pid
进程编号(PID)是Linux内核为每个运行中的进程分配的唯一数字标识。通过PID,用户可以精准定位目标进程并对其进行管理。以下命令可用于快速获取当前系统中的进程信息:
查看进程此处我用的指令是:
ps -a
则会显示出如下图所示结果:

2、命令终止一个进程pid
当进程出现异常或需要主动释放资源时,可通过发送终止信号(Signal)强制结束进程。Linux提供多种信号类型,默认使用SIGTERM(15)请求进程正常退出,若未响应可升级至SIGKILL(9)强制终止。
此处要达到终止一个进程的效果,我们可以先创建一个新的进程,再强制终止,步骤如下:
- 启动测试进程:
sleep 300 & # 后台运行一个休眠300秒的进程
- 查找其PID:
ps -a | grep sleep

- 终止进程
kill 1936

二、初识Linux系统的虚拟机内存管理
(一)虚拟机内存管理
Linux内存管理通过虚拟内存和分页技术为进程提供独立地址空间,由MMU通过页表映射至物理内存或Swap。采用按需分页,访问时触发缺页中断分配内存;内存不足时通过LRU等算法换出非活跃页,由kswapd回收资源,极端时OOM Killer终止进程。写时复制优化进程创建效率。该机制确保进程隔离、内存高效利用及动态扩展能力
(二)与STM32内存管理对比
Linux的虚拟内存管理与STM32的真实物理内存映射在设计理念和应用场景上有显著区别,主要体现在以下几个方面:
-
地址隔离与抽象
Linux通过MMU将虚拟地址映射到物理内存,实现进程间内存隔离;STM32直接操作物理地址,无隔离,需手动管理布局,易因越界操作崩溃。 -
保护与扩展能力
Linux利用页表权限和Swap扩展内存,防御攻击;STM32依赖有限MPU保护关键区,无动态扩展,内存严格受限。 -
实时性与开销
STM32物理内存访问确定、无转换延迟,适合实时控制;Linux可能因页错误或Swap引入延迟,且MMU管理消耗资源。 -
开发模式
Linux自动管理内存,简化开发;STM32需手动分配内存/外设,底层控制强但易出错。
三、Linux调用函数编程
接下来我们将熟悉通过虚拟机在Linux系统中编写c语言程序,熟练调用 fork()、wait()、exec() 等函数。首先我们了解一下上述几个函数的含义:
1、fork( ):创建子进程
-
功能:复制当前进程(父进程),生成一个几乎完全相同的子进程。
-
特点:
-
调用一次,返回两次:父进程返回子进程的 PID(进程标识符),子进程返回
0。 -
子进程继承父进程的代码、数据段、堆栈和文件描述符等资源。
-
2、wait( ) :回收子进程资源
- 功能:父进程阻塞等待子进程终止,并回收其资源
-
特点:
-
wait(NULL) 等待任意子进程结束;waitpid( ) 可指定等待特定子进程。
-
通过参数获取子进程退出状态。
-
3、exec( )执行新程序
-
功能:加载并运行一个新的可执行程序,替换当前进程的代码和数据。
-
特点:
-
属于函数族,参数传递方式不同。
-
调用成功后,原进程的代码段、数据段等被新程序完全覆盖,但进程 PID 不变。
-
具体操作:
1、打开XTerminal
登录我们老师分配的阿里云服务器Ubuntu系统的账号,进入终端


2、通过命令创建Homework文件夹
mkdir ~/homework && cd ~/homework

3、在homework文件夹中通过vi命令创建C语言文件并写入测试代码
vi example.c

测试代码:
#include <unistd.h>
#include <sys/wait.h>
#include <stdio.h>
#include <stdlib.h>int main() {pid_t pid = fork();if (pid == -1) {perror("fork失败");exit(EXIT_FAILURE);} else if (pid == 0) {// 子进程执行ls -lprintf("子进程PID: %d\n", getpid());execl("/bin/ls", "ls", "-l", NULL);// 若execl失败,执行以下代码perror("execl失败");exit(EXIT_FAILURE);} else {// 父进程等待子进程printf("父进程,等待子进程PID: %d\n", pid);int status;wait(&status);if (WIFEXITED(status)) {printf("子进程退出码: %d\n", WEXITSTATUS(status));}printf("父进程结束\n");}return 0;
}
代码分析:
(1)使用 fork( ) 创建子进程
pid_t pid = fork();
- 调用fork( )创建一个新进程(子进程)。
(2)错误处理:fork失败
if (pid == -1) {perror("fork失败");exit(EXIT_FAILURE);
}
-
如果fork( ) 失败(返回
-1),打印错误信息并终止程序。
(3)子进程逻辑
else if (pid == 0) {// 子进程代码printf("子进程PID: %d\n", getpid());execl("/bin/ls", "ls", "-l", NULL);perror("execl失败");exit(EXIT_FAILURE);
}
- 子进程通过getid( )获取自身PID并打印,随后调用execl("/bin/ls","ls","-l",NULL)执行ls -1命令,若路径或参数错误则通过perror( )输出错误信息并调用exit(1)终止自身。
(4)父进程逻辑
else {// 父进程代码printf("父进程,等待子进程PID: %d\n", pid);int status;wait(&status);if (WIFEXITED(status)) {printf("子进程退出码: %d\n", WEXITSTATUS(status));}printf("父进程结束\n");
}
- 父进程通过
wait(&status)阻塞等待子进程终止,检查其是否正常退出(WIFEXITED),获取退出码(WEXITSTATUS),最后打印父进程结束信息。
4、对编辑好的文件保存并退出
在下方的对话框中输入如下指令则可进行对应操作
(1)保存文件:
:w
- :表示进入命令输入状态。
- w 表示写入(write),即保存文件。
(2)退出 vi
:q
- q表示退出(quit)。
- 按 ctrl+enter 退出 vi。
5、编译并运行
回到终端界面输入以下命令即可编译并运行
gcc example.c -o example
./example
结果:

四、总结
在本次学习中,我深入理解了Linux的进程与线程概念,掌握了通过ps、kill等命令查看和终止进程的操作,并通过fork()、wait()、exec()函数实现父子进程协作的编程实践。对比Linux虚拟内存与STM32物理内存管理机制,我认识到前者通过隔离与动态扩展提升安全性与灵活性,而后者以实时性和确定性服务于嵌入式场景。通过编写C程序调用系统函数,我进一步熟悉了多进程资源管理及错误处理流程,巩固了理论与实践的衔接能力,为后续复杂系统开发奠定了基础。
相关文章:
嵌入式——Linux系统的使用以及编程练习
目录 一、Linux的进程、线程概念 (一)命令控制进程 1、命令查看各进程的编号pid 2、命令终止一个进程pid 二、初识Linux系统的虚拟机内存管理 (一)虚拟机内存管理 (二)与STM32内存管理对比 三、Lin…...
(回滚莫队)洛谷 P10268 符卡对决 题解
居然还没调出来?感觉是数据类型的问题,真是吓人。先把思路写一下吧。 题意 灵梦一共有 n n n 张符卡,每张卡都有一个能力值,对于第 i i i 张卡,它的能力值为 a i a_i ai,现在她想从中选出两张符卡并…...
在MacOS 10.15上使用MongoDB
这次是在MacOS 10.15上使用MongoDB。先在豆包问支持MacOS 10.15的MongoDB最新版是什么,答案是MongoDB 5.0。 抱着谨慎怀疑的态度去官方网站查询了一下,答案如下 MongoDB 7.x支持的最低版本MacOS是11MongoDB 6.x支持的最低版本MacOS是10.14 又找deepsee…...
思二勋:未来所有的业务都将生于AI、长于AI、成于AI
每个时代都有其标志性的技术,每个技术的产生或极大地解放了个体的劳动力,提高了个体与组织之间的协作效率,或极大地促进了生产效率或使用体验,或将极大地优化了资源配置和供需匹配效率,从而提高人们的生活水平。从青铜…...
混合专家模型(MoE):助力大模型实现高效计算
引言 近年来,大模型的参数规模不断攀升,如何在保证性能的前提下降低计算成本和显存消耗,成为业界关注的重点问题。混合专家模型(Mixture of Experts, MoE)应运而生,通过“分而治之”的设计理念,…...
【学习笔记】计算机网络(七)—— 网络安全
第7章 网络安全 文章目录 第7章 网络安全7.1 网络安全问题概述7.1.1 计算机网络面临的安全性威胁7.1.2 安全的计算机网络7.1.3 数据加密模型 7.2 两类密码体制7.2.1 对称密钥密码体制7.2.2 公钥密码体制 7.3 鉴别7.3.1 报文鉴别7.3.2 实体鉴别 7.4 密钥分配7.4.1 对称密钥的分配…...
预测分析(四):面向预测分析的神经网络简介
文章目录 面向预测分析的神经网络简介神经网络模型1. 基本概念2. 前馈神经网络3. 常见激活函数4. 循环神经网络(RNN)5. 卷积神经网络(CNN) MPL结构工作原理激活函数训练方法 基于神经网络的回归——以钻石为例构建预测钻石价格的M…...
Debezium日常分享系列之:Debezium 3.1.0.Final发布
Debezium日常分享系列之:Debezium 3.1.0.Final发布 重大改变Debezium Core事件源块现在带有版本号稀疏向量逻辑类型重命名更改了模式历史配置的默认值 Debezium Storage moduleJDBC 存储配置命名约定变更 Debezium for Oracle多个 Oracle LogMiner JMX 指标被移除重…...
LLaMA-Factory大模型微调全流程指南
该文档为LLaMA-Factory大模型微调提供了完整的技术指导,涵盖了从环境搭建到模型训练、推理和合并模型的全流程,适用于需要进行大模型预训练和微调的技术人员。 一、docker 容器服务 请参考如下资料制作 docker 容器服务,其中,挂…...
为什么芯片半导体行业需要全星APQP系统?--行业研发项目管理软件系统
为什么芯片半导体行业需要全星APQP系统?--行业研发项目管理软件系统 在芯片半导体行业,严格的合规性要求、复杂的供应链协同及高精度质量管理是核心挑战。全星研发项目管理APQP系统专为高门槛制造业设计,深度融合APQP五大阶段(从设…...
Linux make 检查依赖文件更新的原理
1. 文件的时间戳 make 主要依靠文件的时间戳来判断依赖文件是否有更新。每个文件在文件系统中都有一个时间戳,记录了文件的三种重要时间: 访问时间(Accesstime):文件最后一次被访问的时间。修改时间&…...
vulkanscenegraph显示倾斜模型(5.6)-vsg::RenderGraph的创建
前言 上一章深入分析了vsg::CommandGraph的创建过程及其通过子场景遍历实现Vulkan命令录制的机制。本章将在该基础上,进一步探讨Vulkan命令录制中的核心封装——vsg::RenderGraph。作为渲染流程的关键组件,RenderGraph封装了vkCmdBeginRenderPass和vkCmd…...
解锁 Python 多线程的潜力:全局解释器锁(GIL)深度解析与优化之道
解锁 Python 多线程的潜力:全局解释器锁(GIL)深度解析与优化之道 引言 Python,这门以简洁和优雅著称的编程语言,自诞生以来在 Web 开发、数据分析、人工智能等领域大放异彩。然而,Python 的多线程性能却常被诟病,其核心原因之一便是全局解释器锁(Global Interpreter …...
基于阿里云可观测产品构建企业级告警体系的通用路径与最佳实践
前言 1.1 日常生活中的告警 任何连续稳定运行的生产系统都离不开有效的监控与报警机制。通过监控,我们可以实时掌握系统和业务的运行状态;而报警则帮助我们及时发现并响应监控指标及业务中的异常情况。 在日常生活中,我们也经常遇到各种各样…...
二叉树的ACM板子(自用)
package 二叉树的中序遍历;import java.util.*;// 定义二叉树节点 class TreeNode {int val; // 节点值TreeNode left; // 左子节点TreeNode right; // 右子节点// 构造函数TreeNode(int x) {val x;} }public class DMain {// 构建二叉树(层序遍历方式&…...
架构思维:查询分离 - 表数据量大查询缓慢的优化方案
文章目录 Pre引言案例何谓查询分离?何种场景下使用查询分离?查询分离实现思路1. 如何触发查询分离?方式一: 修改业务代码:在写入常规数据后,同步建立查询数据。方式二:修改业务代码:…...
Qt进阶开发:QFileSystemModel的使用
文章目录 一、QFileSystemModel的基本介绍二、QFileSystemModel的基本使用2.1 在 QTreeView 中使用2.2 在 QListView 中使用2.3 在 QTableView 中使用 三、QFileSystemModel的常用API3.1 设置根目录3.2 过滤文件3.2.1 仅显示文件3.2.2 只显示特定后缀的文件3.2.3 只显示目录 四…...
后端开发常见的面试问题
目录 编程语言 python Linux环境 web框架 数据处理与分析 数据库 图数据库 什么是图数据库?它与传统关系型数据库有什么区别? 图数据库中的节点、边和属性分别代表什么? 常见的图数据库有哪些?它们各自有什么特点&#…...
List结构之非实时榜单实战
像京东、淘宝等电商系统一般都会有热销的商品榜单,比如热销手机榜单,热销电脑榜单,这些都是非实时的榜单。为什么是非实时的呢?因为完全实时的计算和排序对于资源消耗较大,尤其是当涉及大量交易数据时。 一般来说&…...
【C语言】字符串处理函数:strtok和strerror
在C语言中,字符串处理是编程的基础之一。本文将详细讲解两个重要的字符串处理函数:strtok和strerror 一、strtok函数 strtok函数用于将字符串分割成多个子串,这些子串由指定的分隔符分隔。其原型定义如下: char *strtok(char *s…...
如何提升后端开发效率:从Spring Boot到微服务架构
在现代软件开发中,后端开发的效率直接决定了项目的成败。随着技术的快速发展,Spring Boot、微服务架构、Docker等工具和技术已经成为提升后端开发效率的核心利器。在这篇文章中,我们将探讨如何通过使用Spring Boot及微服务架构来提升开发效率…...
go语言:开发一个最简单的用户登录界面
1.用deepseek生成前端页面: 1.提问:请你用html帮我设计一个用户登录页面,要求特效采用科技感的背景渲染加粒子流动,用css、div、span标签,并给出最终合并后的代码。 生成的完整代码如下: <!DOCTYPE h…...
基于 .NET 8 + Lucene.Net + 结巴分词实现全文检索与匹配度打分实战指南
文章目录 前言一、技术选型与优势1.1 技术栈介绍1.2 方案优势 二、环境搭建与配置2.1 安装 NuGet 包2.2 初始化核心组件 三、索引创建与文档管理3.1 构建索引3.2 动态更新策略 四、搜索与匹配度排序4.1 执行搜索4.2 自定义评分算法(扩展) 五、高级优化技…...
Docker安装、配置Nacos
1.如果没有docker-compose.yml文件的话,先创建docker-compose.yml 配置文件一般长这个样子 version: 3services:nacos:image: nacos/nacos-server:v2.1.1container_name: nacos2ports:- "8848:8848"- "9848:9848"environment:- MODEstandalone…...
《Maven高级应用:继承聚合设计与私服Nexus实战指南》
一、 Maven的继承和聚合 1.什么是继承 Maven 的依赖传递机制可以一定程度上简化 POM 的配置,但这仅限于存在依赖关系的项目或模块中。当一个项目的多个模块都依赖于相同 jar 包的相同版本,且这些模块之间不存在依赖关系,这就导致同一个依赖…...
重要头文件下的函数
1、<cctype> #include<cctype>加入这个头文件就可以调用以下函数: 1、isalpha(x) 判断x是否为字母 isalpha 2、isdigit(x) 判断x是否为数字 isdigit 3、islower(x) 判断x是否为小写字母 islower 4、isupper(x) 判断x是否为大写字母 isupper 5、isa…...
C语言数字分隔题目
一、题目引入 编写一个程序,打印出从用户输入的数字开始,递减到1的序列。要求每次打印一行,数字之间用逗号分隔,最后一个数字后面没有逗号。 二、代码展示 三、运行结果 四、思路分析 1.先用一个for循环对输入的数字进行递减 2.再对for循环里面的数字进行筛选 如果大于1 …...
DigitalOcean 发布 AMD Instinct MI300X GPU 裸金属服务器
DigitalOcean 宣布现已提供 AMD Instinct MI300X GPU,并搭载 ROCm 软件,以支持用户的 AI 任务。 在 DigitalOcean,我们致力于为你的项目提供更多选择。AMD Instinct MI300X 是目前带宽最高的 GPU 之一(5.3 TB/s 的 HBM3 内存带宽&…...
CentOS 7 镜像源失效解决方案(2025年)
执行 yum update 报错: yum install -y yum-utils \ > device-mapper-persistent-data \ > lvm2 --skip-broken 已加载插件:fastestmirror, langpacks Loading mirror speeds from cached hostfile Could not retrieve mirrorlist http://mirror…...
应对高并发的根本挑战:思维转变【大模型总结】
以下是对这篇技术总结的详细解析,以分步说明的形式呈现,帮助理解亿万并发场景下的核心策略与创新思维: 一、应对高并发的根本挑战:思维转变 1. 传统架构的局限 问题:传统系统追求零故障和强一致性,但在海…...
