Linux进程控制
进程控制
- fork函数
- 进程终止
- 退出码
- 常见的退出方式
- 进程等待
- 什么是进程等待,为什么要进程等待
- 阻塞与非阻塞
- 进程替换
- 替换原理
- 替换函数
- 执行系统命令
- 执行自己写的程序
- 模拟实现简易的shell
fork函数
fork函数是创建一个子进程,之前用过。
#include <unistd.h>
pid_t fork(void);
返回值:自进程中返回0,父进程返回子进程id,出错返回-1。
进程拥有独立性,fork之后就变成了两个程序,父子进程共享后边的代码。
那么为什么给父进程返回的就是子进程的pid,而给子进程返回的就是0呢?
就好比孩子只能有一个亲生的父亲,而一个父亲可以拥有很多亲生孩子,每个孩子都是独立不同的。
fork函数是在什么时候创建的子进程呢?
pid_t fork()
{1.创建PCB2.赋值3.创建进程地址空间4.赋值5.创建并设置页表6.子进程放入进程队列//这里才是创建成功一个进程,也是分流的地方7.........return pid;//返回的时候核心代码已经执行完毕了
}
也就是说fork返回两个值是因为返回之前就已经创建好新进程了。
返回的本质就是写入,谁先返回谁先写入id,因为进程的独立性,然后就会发生写时拷贝。
fork失败的原因
系统拥有太多个进程超过了用户进程的限制就会失败。
进程终止
退出码
在写C/C++的时候,我们在main函数是程序的开始,但是最后一个位置会写return 0;
这也就代表一个程序的退出,至于为什么要写return 0,而不是返回其他的,亦或者是不写都可以,因为返回uid这个数字是退出码,0是正常退出的意思,因为正确只有一个,不会管你怎么成功,但是失败就会找失败的原因再去改正。
echo $?是查看最近进程的退出码,上一个写的进程退出码是1,再查一次就是echo $?的退出码,是0.
退出码可以自定义,也可以使用系统的映射关系,这里不太推荐。
这个之前用过:
然后来看看里面数字对应的错误信息
注意:如果程序异常退出码也无意义。
常见的退出方式
上面说了在mian函数中调用return就是进程退出。
C语言和操作系统还提供了两个函数退出进程:
这是C语言提供的一个函数,只要使用就会退出当前进程,参数是退出码。
无论是在哪个位置,或者是后面有多少代码。
还有一个系统级别调用的是_exit,作用几乎相同:
系统调用的并没有打印。
这是C语言提供的,过了两秒钟就打印出来了。
这说明:
exit 终止进程后会主动刷新缓冲区。
_exit 终止之后不会主动刷新缓冲区。
那么这个缓冲区在哪里呢?
exit会刷新缓冲区,但是系统不会,也就是说位置在系统调用和库函数之间,具体的以后说。
进程等待
什么是进程等待,为什么要进程等待
之前说过僵尸进程会导致内存泄漏,因为他的资源无法回收,所以就需要等待子进程结束然后来保存资源给父进程,通过获取子进程退出信息知道是否成功退出。
首先来看两个等待进程的函数。
wait/waitpid:
status参数是拿该进程的退出结果。
options参数是传入阻塞和非阻塞状态。
pid_t是返回进程的pid,返回-1代表失败。
wait
这个程序10-15秒是僵尸进程,15s之后就会被回收,这个时候子进程就不是僵尸状态了。
waitpid
在举例之前首先说一下status:
一个程序终止有三种情况,代码运行完毕,结果正确和不正确,还有没运行完,出异常了。
这个时候status是获取他们这个信息的,并且它是拥有自己的位图结构的。
一共有32个比特位,其中重要的只有16个比特位:
终止信号是一个进程出异常了会受到终止信号,暂时用来判断进程是否正常退出。
退出状态是看结果是否正确。
这个是等待的过程,其实就是status去PCB找信号和退出码。
总结来说:status让操作系统释放掉僵尸状态,然后获取进程的退出结果。
但是如果让我们自己去求信号和退出码很麻烦,所以Linux提供了一些操作的宏,重点说两个:
WIFEXITED(status): 若为正常终止子进程返回的状态,则为真。(查看进程是否是正常退出)
WEXITSTATUS(status): 若WIFEXITED非零,提取子进程退出码。(查看进程的退出码)
所以就可以改良成这个样子:
结论:
进程退出会变成僵尸,之后将自己的推出结果放入PCB,wait/waitpid是系统调用,有资格去读取PCB中的资源。
阻塞与非阻塞
阻塞
父进程一直在等子进程结束回收资源。
非阻塞
父进程一段时间过来看一下子进程是否结束,如果没结束可以做其他事情,这个叫轮询方式。
WNOHANG: 若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若正常结束,则返回该子进程的ID。
NO1就父进程是不是询问子进程是否退出。
如果在询问之后不子进程没有准备完毕,父进程则可以做一些其他的事情。
至于非阻塞和阻塞谁更好,这个要看实际场景。
进程替换
之前说过创建子进程的目的是让子进程去帮忙“做事”,可是为什么要去让子进程帮忙做事呢?
首先说目的:
1.想让子进程执行父进程磁盘代码其中的一部分。
2.想让子进程执行一个全新的程序。
替换原理
一个可执行程序被首先被加载到内存中,然后执行代码,然后代码中有操作让本程序执行一个新程序,这个时候就会将指定执行的程序的代码和数据覆盖掉原本的代码和数据,在整个过程中并没有产生新的进程,这就是为什么每次都要去创建一个子进程来去执行新程序。
替换函数
执行系统命令
这些函数的作用是将指定的程序加载到内存当中,让指定的进程执行。
int execl(const char *path, const char *arg, …);
第一个参数是说如何找到程序,第二个参数是我们在命令行解释器怎么调用该程序就怎么写,最后用NULL结尾。
这几个函数统一的是exec,这个函数最后一个l 意思是 list 将参数一个一个传入exec*
这里执行完execl之后,后边打印process就不会执行,因为整个程序的代码和数据已经被覆盖掉了。
并且这类函数返回值只有-1,表示错误。
因为成功之后接下来的代码是不会执行的,所以返回一个正确的值进行判断也毫无意义。
int execlp(const char *file, const char *arg, …);
结尾是p的第一个参数不用去指定路径了,他会在环境变量PATH,进行可执行程序的查找
int execv(const char *path, const char *argv[], …);
v是vector的意思,第二个参数是让我们把所有可执行参数放入数组中传过去。
int execvp(const char *file, const char *argv[], …);
这个就不演示了,这两个参数上面都说过。
上面的只是在执行系统命令,那么想执行自己写的程序该怎么办呢?
执行自己写的程序
首先来说一下makefile这个文件:
先创建一个.c文件
如果我想让test.c去调用process.c,首先要生成这两个可执行程序,但是makefile只会默认的生成第一个可执行程序,后面的就不会再去执行,所以我们要这样写:
因为是调用程序,所以不管是什么语言的程序都可调用。
int execle(const char *path, const char *arg, …,char *const envp[]);
最后一个参数是自定义环境变量的意思。
现在的自定义环境变量还没定义,所以为空。
我们发现,如果没有自定义环境变量,系统自带的环境变量就会被打印,但是如果自定义环境变量系统自带的环境变量就不会被打印。
那么如果我两个都想要怎么办呢?
这个函数传入你的自定义环境变量就可以了,作用就是将你定义的环境变量导入到系统当中。
这里穿插一个问题,一个程序运行之前,是先调用main还是先调用exec函数呢?
是先调用exec函数,因为它的作用上面说了,是将程序加载到内存中,Linux中,它就是加载器。
调用exec函数之后会将自己的参数等等传给main函数,这就是为什么之前说main函数有三个参数,谁传给他的。
int execvpe(const char *file, const char *argv[], …,char *const envp[]);
这个参数就不说了,都说过了。
注意:上面这些接口都是execve系统调用,其他的都是封装,为了让我们有更好的选择性。
模拟实现简易的shell
首先来利用main函数的参数来实现一个功能:
那么我们可有利用这个模拟实现一个简单的shell。
第一步先设置输入和输出,并且创建一个字符数组储存输入的参数。
我们输入一个字符串是abc,然后会按回车,也就是说实际上是abc\n,如果我要在打印信息%s后面加一个\n那么就会多出一行,不加容易出现缓冲区不刷新问题,所以我们要去除输入末尾的\n。
第二步要进行字符串分割,因为我们在屏幕输入的是ls -a -l这种,但是exec函数要用到的是字符指针数组类型的,所以我们创建一个字符指针数组,然后进行分割放进字符指针数组:
这里要说一下内建命令,我们在输入ls什么的时候不同文件会有颜色,但是如果调用exec里面就需要自己添加颜色选项,我们又不能在屏幕输入,所以只能在代码中添加,首先判断一定要是ls命令才行,然后添加颜色选项。
像这种不需要让子进程来执行,而是shell自己执行的就叫做内建命令。
第三步是打印,创建一个子进程帮我们工作,这是因为exec函数会替换掉原来程序中所有的代码和数据:
然后我们还可以设置一个条件编译来看看字符指针数组中的字符切割是否正确:
先来测试一下上面的程序是否正确
但是如果我们输入cd …就会发现根本没有任何变化,这是为什么呢?
先创建一个其他程序来看一下一个进程的状态:
用ls /proc/pid -al
cwd是当前进程的工作目录,也是我们平时说的当前路径,exe是当前程序执行的是磁盘路径的哪一个程序。
那么这个当前路径可以改变嘛?通过一个函数是可以的:
谁调用这个函数就更改谁的工作目录,参数是更改到哪个目录。
如果更改了工作目录,那么以后这个程序再进行创建文件等等操作,就会再新的工作目录创建,因为系统默认是跟可执行程序同一个目录下去创建新文件。
那么刚才我们的shell不能cd …是因为他只能让当前工作目录发生变化,因为shell是通过创建子进程去执行命令,我们让目录进行变化的时候是让子进程去帮助执行,也就是说改变的其实是子进程的目录,和父进程没有任何关系,所以说这里还需要创建一个内建命令:
之有前还有一个命令,是echo $?,返回的是最近一次退出码
首先创建两个全局变量保存退出码和信号,然后再用他们储存子进程返回的结果:
最后进行判断:
这里简单的完善一下就可以了,主要是综合了上面所说的大部分内容。
相关文章:

Linux进程控制
进程控制fork函数进程终止退出码常见的退出方式进程等待什么是进程等待,为什么要进程等待阻塞与非阻塞进程替换替换原理替换函数执行系统命令执行自己写的程序模拟实现简易的shellfork函数 fork函数是创建一个子进程,之前用过。 #include <unistd.h…...

PMP项目管理引论介绍
目录1. 指南概述和目的1.1 项目管理标准1.2 道德与专业行为规范2 基本要素2.1 项目2.2 项目管理的重要性2.3 项目、项目集、项目组合以及运营管理之间的关系2.3.1 概述2.3.2. 项目组合与项目集管理2.3.3. 运营管理2.3.4. 组织级项目管理和战略2.3.5. 项目管理2.3.6. 运营管理与…...

计算机视觉废钢堆提取问题
计算机视觉废钢堆提取问题 背景介绍 在钢铁炼制中,废钢是非常重要的原料,不同等级废钢对于钢成品影响很大,因此需要对废钢进行正确分类。某废钢料场中,卸料区域布置了多个摄像头,用于拍摄卸料场中废钢堆,…...

判断水仙花数-课后程序(Python程序开发案例教程-黑马程序员编著-第二章-课后作业)
实例5:判断水仙花数 水仙花数是一个3位数,它的每位数字的3次幂之和等于它本身,例如13 53 33 153,153就是一个水仙花数。 本实例要求编写程序,实现判断用户输入的3位数是否为水仙花数的功能。 实例目标 掌握Pytho…...

目标检测: 数据增强代码详解
1. 常见的数据增强 1.1 翻转图像 左右水平翻转 假设图片的宽高为w,h,bdbox左上角A坐标为(x1,y1), 右下角B为(x2,y2)。经过左右水平翻转后,bdbox的左上角A1坐标(w-x2,y1) ,右下角B1坐标为(w-x1,y2)左右水平翻转的代码实现如下:from PIL import Image image = Image.open(i…...

第二讲:ambari编译复盘,如何实现一次性成功编译ambari
上节课我们已经讲解了如何成功编译ambari源码,安装ambari-server rpm包以及成功部署ambari。本节课我们来复盘一下上节课的编译过程,以及思考如何实现一次性成功编译ambari。 要想一次性成功编译ambari,那么就需要将预置工作做好,比如: maven镜像源配置,node_moudle模块…...

Windows下jdk安装与卸载-超详细的图文教程
jdk安装 下载jdk 由于现在主流就是jdk1.8,所以这里就下载jdk1.8进行演示。官方下载地址:https://www.oracle.com/java/technologies/downloads/#java8-windows。 官方下载需要注册oracle账号,国内下载有可能速度慢,若不想注册账…...

Jackson CVE-2018-5968 反序列化漏洞
0x00 前言 同CVE-2017-15095一样,是CVE-2017-7525黑名单绕过的漏洞,主要还是看一下绕过的调用链利用方式。 可以先看: Jackson 反序列化漏洞原理 或者直接看总结也可以: Jackson总结 影响版本:至2.8.11和2.9.x至…...

解析MySQL 8.0 OCP(1Z0-908)考试中一道大部分同学都会做错的题目
一个用户有下面的权限: mysql>SHOW GRANTS FOR jsmith;---------------------------------------------------------------------- | Grants for jsmith% | ----------------------------------------------------------…...

Java死锁
什么是死锁? 多个线程同时被阻塞,它们中的一个或者全部都在等待某个资源被释放。由于线程被无限期地阻塞,因此程序不可能正常终止。 死锁的必要条件: 1、互斥条件:该资源任意一个时刻只由一个线程占用。 2、请求与…...

BloomFilter原理学习
文章目录BloomFilter简单介绍BloomFilter中的数学知识fpp(误判率/假阳性)的计算k的最小值公式总结编程语言实现golang的实现[已知n, p求m和k](https://github.com/bits-and-blooms/bloom/blob/master/bloom.go#L133)参考BloomFilter简单介绍 BloomFilter我们可能经常听到也在使…...

C语言老题新解第1-5题
文章目录1 互不相同且无重复数字2 企业利润提成3 两个完全平方数4 判断一年的第几天5 三个整数比较大小1 互不相同且无重复数字 1 有1, 2, 3, 4四个数字,能组成多少互不相同且无重复数字的三位数?都是多少? 最简单当然是三重循环嵌套在一起…...

【数据库系列】MQSQL历史数据分区
互联网行业企业都倾向于mysql数据库,虽说mysql单表能支持亿级别的数据量,加上索引优化下查询速度,勉强能使用,但是对于追求性能和效率的互联网企业,这是远远不够的。Mysql数据库单表数据量到达500万左右,达…...

MyBatis常用的俩种分页方式
1、使用 limit 实现分页 select * from xxx limit m,n # m 表示从第几条数据开始,默认从0开始 # n 表示查询几条数据 select * from xxx limit 2,3 # 从索引为2的数据开始,往后查询三个。2、3、4 (1) 创建分页对象,用来封装分页的数据 PS…...

RPC通信原理解析
一、什么是RPC框架? RPC,全称为Remote Procedure Call,即远程过程调用,是一种计算机通信协议。 比如现在有两台机器:A机器和B机器,并且分别部署了应用A和应用B。假设此时位于A机器上的A应用想要调用位于B机…...

【蓝桥杯集训·周赛】AcWing 第93场周赛
文章目录第一题 AcWing 4867. 整除数一、题目1、原题链接2、题目描述二、解题报告1、思路分析2、时间复杂度3、代码详解第二题 AcWing 4868. 数字替换一、题目1、原题链接2、题目描述二、解题报告1、思路分析2、时间复杂度3、代码详解第三题 AcWing 4869. 异或值一、题目1、原题…...

蓝桥杯-刷题统计
蓝桥杯-刷题统计1、问题描述2、解题思路3、代码实现3.1 方案一:累加方法(超时)3.2 方案二1、问题描述 小明决定从下周一开始努力刷题准备蓝桥杯竞赛。他计划周一至周五每天做 a 道题目, 周六和周日每天做 b 道题目。请你帮小明计算, 按照计划他将在 第几天实现做题数…...

Linux入门教程||Linux Shell 变量|| Shell 传递参数
Shell 变量 定义变量时,变量名不加美元符号($,PHP语言中变量需要),如: your_name"w3cschool.cn"注意,变量名和等号之间不能有空格,这可能和你熟悉的所有编程语言都不一…...

[算法和数据结构]--回溯算法之DFS初识
回溯算法——DFSDFS介绍(Depth First Search)DFS经典题目1. 员工的重要性2. 图像渲染3.被围绕的区域4.岛屿数量5. 电话号码的字母组合6.数字组合7. 活字印刷8. N皇后DFS介绍(Depth First Search) 回溯法(back tracking)(探索与回溯法&#x…...

【LeetCode每日一题】——680.验证回文串 II
文章目录一【题目类别】二【题目难度】三【题目编号】四【题目描述】五【题目示例】六【解题思路】七【题目提示】八【时间频度】九【代码实现】十【提交结果】一【题目类别】 贪心算法 二【题目难度】 简单 三【题目编号】 680.验证回文串 II 四【题目描述】 给你一个字…...

【C语言进阶:指针的进阶】你真分得清sizeof和strlen?
本章重点内容: 字符指针指针数组数组指针数组传参和指针传参函数指针函数指针数组指向函数指针数组的指针回调函数指针和数组面试题的解析这篇博客 FLASH 将带大家一起来练习一些容易让人凌乱的题目,通过这些题目来进一步加深和巩固对数组,指…...

【前端必看】极大提高开发效率的网页 JS 调试技巧
大家好,我是前端西瓜哥。本文讲解如何使用浏览器提供的工具进行 JS 代码的断点调试。 debugger 在代码中需要打断点的地方,加上 debugger,表示一个断点。浏览器代码执行到该位置时,会停下来,进入调试模式。 示例代码…...

【春招面经】视源股份前端一面
前言 本次主要记录一下视源股份CVTE前端一面 (3.3下午4点15) 文章目录前言本次主要记录一下视源股份CVTE前端一面 (3.3下午4点15)问题总结介绍一下项目的来源以及做这个项目的初衷一直监听滚动,有没有对性能产生影响&a…...

插件化开发入门
一、背景顾名思义,插件化开发就是将某个功能代码封装为一个插件模块,通过插件中心的配置来下载、激活、禁用、或者卸载,主程序无需再次重启即可获取新的功能,从而实现快速集成。当然,实现这样的效果,必须遵…...

tftp、nfs 服务器环境搭建
目录 一、认识 tftp、nfs 1、什么是 tftp? 2、什么是 nfs? 3、tftp 和 nfs 的区别 二、tftp的安装 1、安装 tftp 服务端 2、配置 tftp 3、启动 tftp 服务 三、nfs 的安装 1、安装 nfs 服务端 2、配置 nfs 3、启动 nfs 服务 一、认识 tftp、…...

汇编系列03-不借助操作系统输出Hello World
每天进步一点点,加油! 上一节,我们通过汇编指令,借助操作系统的系统调用实现了向标准输出打印Hello world。这一节我们打算绕过操作系统,直接在显示屏幕上打印Hello world。 计算机的启动过程 当我们给计算机加电启…...

TPU编程竞赛系列|算能赛道冠军SO-FAST团队获第十届CCF BDCI总决赛特等奖!
近日,第十届中国计算机学会(CCF)大数据与计算智能大赛总决赛暨颁奖典礼在苏州顺利落幕,算能赛道的冠军队伍SO-FAST从2万余支队伍中脱颖而出,获得了所有赛道综合评比特等奖! 本届CCF大赛吸引了来自全国的2万…...

【C++】AVL树,平衡二叉树详细解析
文章目录前言1.AVL树的概念2.AVL树节点的定义3.AVL树的插入4.AVL树的旋转左单旋右单旋左右双旋右左双旋AVL树的验证AVL树的删除AVL树的性能前言 前面对map/multimap/set/multiset进行了简单的介绍,在其文档介绍中发现,这几个容器有个共同点是࿱…...

C/C++开发,无可避免的多线程(篇四).线程与函数的奇妙碰撞
一、函数、函数指针及函数对象 1.1 函数 函数(function)是把一个语句序列(函数体, function body)关联到一个名字和零或更多个函数形参(function parameter)的列表的 C 实体,可以通过返回或者抛…...

elisp简单实例: taglist
从vim 转到emacs 下,一直为缺少vim 中的tablist 插件而感到失落. 从网上得到的一个emacs中的taglist, 它的功能很简陋,而且没有任何说明, 把它做为elisp的简单实例,供初学者入门倒不错,我给它加了很多注释,帮助理解, 说实话,感觉这百行代码还是挺有深度的,慢慢体会,调试才会有收…...