一文让你深入程度的了解Linux进程「建议小白收藏」
nanshan 2024-10-24 11:42 11 浏览 0 评论
前言:进程是什么?在操作系统中,我们经常能听到这样的话。我们要终止一个进程或者杀死一个进程,父进程创建了子进程这一类的话。往往我们听到都会觉得很高大上,这跟编程语言完全不同的感觉,操作了整个计算机。
一,进程的概念
冯诺依曼体系结构
在进程之前首先要提一下我们的“祖师爷”——冯诺依曼体系结构。
这个是一个计算机入门第一节课必然会提到的知识。冯诺依曼体系结构提出了计算机采用二进制;计算机应该按照程序顺序执行。它由输入设备,输出设备,存储器,控制器,运算器组成。
【注意!!注意!!!】
1、这里的存储器指的是内存。
2、不考虑缓存情况,这里的cpu(控制器+运算器)只能对内存进行读写,不能访问外设(输入或输出设备)
3、外设要输入或输出数据,也只能写入内存或者从内存中读取
二,操作系统(0S)
任何计算机系统都包含一个基本的程序集合,称为操作系统(OS),操作系统的存在就是让计算机更加的好用,能更方便的、统筹合理的管理计算机的软硬件资源。
那么是怎么管理的?(我们可以先举一个例子,学校的管理,首先我们被辅导员统一管理,辅导员又由院内领导管理,院内领导又由校长管理。)
学生->辅导员->院内领导->校长 这么一个层次结构。但在这之前需要制定一个制度来管理,每一个层次的人都遵守这个制度。这样才能按部就班地进行工作的分配。
那么操作系统也一样。总的来说就是:先描述,再组织;描述用struct结构体,比如进程有task_struct这样一个结构体来描述,组织可以用链表或者其他高效的数据结构。
三,系统调用和库函数概念
在开发角度,操作系统对外会表现一个整体,但是会暴露自己的部分接口,供上层开发使用,这部分由操作系统提供的接口,叫做系统调用。
系统调用在使用上,功能比较基础,对用户的要求也相对比较高,所以,有心的开发者可以对部分系统调用进行适度封装,从而形成了库,有了库,就很有利于更上层用户或者开发者进行二次开发。
(1)进程概念
从用户角度:进程就是一个正在运行中的程序。
操作系统角度:操作系统运行一个程序,需要描述这个程序的运行过程,这个描述通过一个结构体task_struct{}来描述,统称为PCB,因此对操作系统来说进程就是PCB(process control block)程序控制块
进程的描述信息有:标识符PID,进程状态,优先级,程序计数器,上下文数据,内存指针,IO状态信息,记账信息。都需要操作系统进行调度。
那么在Linux操作系统下,怎么查看进程呢
(2)查看进程
输入ls /porc指令即可
前面蓝色数字代表的进程的ID。如果你想查看PID为1的进程信息,你需要查看/porc/1这个文件夹
我们也可以使用ps -ef -aux指令来直接显示进程状态.。
还有getpid()和getppid()这两个函数用来查看当前程序的进程和父进程PID。
int main(){
printf("getpid:%d",getpid());
printf("getppid:%d",getppid());
return 0;
}
(3)进程创建
Linux中非常重要的函数——fork(),它从已存在的进程中创建一个新进程。新进程为子进程,而原进程为父进程。
#include <unistd.h>
pid_t fork(void);
返回值:
父进程:返回值大于0,子进程的pid
子进程:返回值等于0
写这么个程序来初始fork函数
#include <stdio.h>
#include <unistd.h>
int main(){
printf("parent pid:%d\n",getpid());
int a = 100;
pid_t pid = fork();
if(pid < 0){
return -1;
}else if(pid == 0){
a = 20;
printf("child !! pid:%d----a:%d--%p\n",getpid(),a ,&a);
}else{
sleep(1);
printf("parent !! pid:%d----a:%d--%p\n",getpid(), a, &a);
}
printf("nihaoa %d\n",a);
return 0;
}
关于fork函数需要理解,每当调用一次fork函数时,会返回两个两次。一次是在调用进程中(父进程)返回一次,返回值是新派生的进程的进程ID。一次是在子进程中返回,返回值是0,代表当前进程为子进程。如果返回-1,那么则代表在创建子进程的过程中出现了错误。
将上面代码执行之后显示如下:
先返回了子进程的pid,之后再返回了父进程的pid;fork()相当于创建了一个新的子进程,但是拷贝的是fork()函数之后的所有数据,之前的并不会拷贝。在代码之上就可以看到parentpid只打印了一次
总的来说:复制pcb,代码共享,但是子进程并非从头开始,而是从fork()函数之后开始,数据独有。
借用一下网上大佬对fork()的理解:
- (1)一个进程进行自身的复制,这样每个副本可以独立的完成具体的操作,在多核处理器中可以并行处理数据。这也是网络服务器的其中一个典型用途,多进程处理多连接请求。
- (2)一个进程想执行另一个程序。比如一个软件包含了两个程序,主程序想调起另一个程序的话,它就可以先调用fork来创建一个自身的拷贝,然后通过exec函数来替换成将要运行的新程序。
那么创建子进程的意义是什么————压力分摊/干其他工作
(4)进程状态
进程状态一般有:就绪态,阻塞态,运行态
在Linux下:R运行状态,S睡眠状态,D磁盘休眠状态,T停止状态,X死亡状态
这些当我们使用指令ps -aux就可以看到
(5)僵尸进程
在进程状态中有两个比较特殊的存在。僵尸和孤儿
僵尸进程是进程退出后,但是资源没有释放,处于僵死状态的进程。
产生原因:子进程先于父进程退出,操作系统检测到进程的退出,通知父进程,但是父进程这时候正在执行其他操作,没有关注这个通知,这时候操作系统为了保护子进程,不会释放子进程资源,因为子进程的PCB中包含有退出原因。这时候因为既没有运行也没有退出,因此处于僵死状态,成为僵尸进程。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
int main()
{
pid_t pid;
//循环创建子进程
while(1)
{
pid = fork();
if (pid < 0)
{
perror("fork error:");
exit(1);
}
else if (pid == 0)
{
printf("I am a child process.\nI am exiting.\n");
//子进程退出,成为僵尸进程
exit(0);
}
else
{
//父进程休眠20s继续创建子进程
sleep(20);
continue;
}
}
return 0;
}
执行这上面这个程序,子进程中途退出了。
z+这个标志就是僵尸进程的标志。
那么怎么避免僵尸进程的产生?
我们一般处理就是关闭父进程,这样僵尸子进程也随之消失了。
所以我们最好设置进程等待,等待子进程完成了工作,并且通知了父进程之后,在退出。
(6)孤儿进程
孤儿进程与僵尸进程在理解上可以认为相反。
父进程先于子进程退出,父进程退出后,子进程成为后台进程,并且父进程为1号进程。
守护进程:特殊(脱离了与终端的关联+会话的关联)的孤儿进程
(7)进程优先级
优先级:决定资源的优先分配权的等级划分
那么进程为什么具有优先级呢?
为了让操作系统的运行更加的合理——交互式进程(一旦有操作优先处理)和批处理进程(一直处理程序,但对CPU要求不高)
设置优先级,可以使用指令ps -elf先查看进程
可以看到 PRI 和 NI这两个数值
PRI:优先级 NI:nice值
PRI是无法直接调整的,但是可以通过调整nice值来调整优先级的大小
PRI = PRI + NI,但是NI也是有范围的——(-20~19)
指令操作为renice -n size -p pid
运行时操作为nice -n size ./main(可执行文件)
这里稍微提一下,程序在运行时具有并行和并发两种执行。
并行:CPU资源足够,多个程序同时运行
并发:CPU资源不够,多个程序切换调度运行(可以看看我之前的一篇关于操作系统的博客,有关调度方法的介绍)
Linux下指令top指令可以查看进程的优先级,进入top后按“r”–>输入进程PID–>输入nice值
四,环境变量
环境变量是保存系统运行环境参数的变量
环境变量在安装java的过程中,可能接触过,需要进入系统环境变量,然后设置PATH添加java的路径
在Linux下可以通过命令指令自己设置
echo 通过变量名称,查看指定环境变量
env 查看所有环境变量
set 查看环境变量以及临时变量
export 声明一个环境
unset 删除一个临时变量
常见的环境变量:HOME SHELL USER PATH
环境变量的全局特性:在子进程中获取继承于父进程的环境变量信息
三种获取环境变量的参数
main(int argc, char *argc[],char *env[]) 参数获取
extern char **environ; 全局变量获取
char *getenv(const char *env_name) 接口获取
写一个获取变量的demo
#include <stdio.h>
#include <stdlib.h>
W>int main(int argc, char *argv[], char *env[]){
//main函数的参数值是从操作系统命令行上获得的。当我们要运行一个可执行文件时,在DOS提示符下键入文件名,再输入实际参数> 即可把这些实参传送到main的形参中去。
//C:\>可执行文件名 参数 参数……
//argv参数是字符串指针数组,其各元素值为命令行中各字符串(参数均按字符串处理)的首地址
//env:字符指针的数组,每一个元素是指向一个环境变量的字符指针
int i;
for(i = 0;i < argc; i++){
printf("argv[%d]=[%s]\n",i , argv[i]);
}
extern char **environ;
for(i = 0;environ[i] != NULL;i++){
printf("env[%d]=[%s]\n",i, environ[i]);
}
char *ptr = getenv("MYENV");
printf("MYENV:[%s]\n",ptr);
return 0;
}
如果输入export MYENV=“10000”,在运行程序,MYENV将变成MYENV:[10000]。
程序地址空间
地址是什么?地址是内存的编号,指向内存的一块区域
虚拟地址空间:
mm_struct{
long int size
code_start
code_end
data_start
a
data_end
}
上图之中,中间的物理内存地址,两边的为虚拟内存地址。
右边是父进程将虚拟地址通过页表查找到物理内存的地址,这时父进程在运行一个程序。此时父进程又创建了一个子进程(图最右)。子进程也通过页表找到物理内存的地址,这是子进程运行的程序在物理内存的另一个新空间运行。
这是虚拟内存地址使用的过程。它的作用就是保持进程的独立性,通过页表映射物理地址,充分地利用物理地址,增加内存访问控制。
这就是进程有关的知识。总结一下
进程从用户角度和操作系统角度去理解。前者就是一个运行的程序,后者表示运行一个程序,需要描述一个程序的运行过程,通过一个结构体task_struct{}来描述,叫做PCB。对操作系统来说,进程就是PCB
进程的创建需要一个我们必须要掌握的函数——fork()函数,创建一个子进程。并且fork()函数会有两个返回值,在返回的过程中父进程之前的数据不进行拷贝,之后与父进程的运行一致。创建子进程是为了分摊压力/干其他工作
进程的状态有运行态,睡眠状态,磁盘休眠状态,停止状态,死亡状态。每一种状态是容易理解的。但是还有两个特殊的进程
僵尸进程和孤儿进程,前者是进程退出后资源没有释放,操作系统通知父进程,但是父进程此时正在处理其他事情,没有关注子进程的退出通知,系统为了保护资源,没有释放掉,并且在子进程的PCB中也保留了退出原因,此时既没有运行也没有完全退出,处于了僵死状态。但是后者是父进程先于子进程先退出,子进程成为了后台程序,由一号进程接管。
相关推荐
- 服务器数据恢复—Raid5数据灾难不用愁,Raid5数据恢复原理了解下
-
Raid5数据恢复算法原理:分布式奇偶校验的独立磁盘结构(被称之为raid5)的数据恢复有一个“奇偶校验”的概念。可以简单的理解为二进制运算中的“异或运算”,通常使用的标识是xor。运算规则:若二者值...
- 服务器数据恢复—多次异常断电导致服务器raid不可用的数据恢复
-
服务器数据恢复环境&故障:由于机房多次断电导致一台服务器中raid阵列信息丢失。该阵列中存放的是文档,上层安装的是Windowsserver操作系统,没有配置ups。因为服务器异常断电重启后,rai...
- 服务器数据恢复-V7000存储更换磁盘数据同步失败的数据恢复案例
-
服务器数据恢复环境:P740+AIX+Sybase+V7000存储,存储阵列柜上共12块SAS机械硬盘(其中一块为热备盘)。服务器故障:存储阵列柜中有磁盘出现故障,工作人员发现后更换磁盘,新更换的磁盘...
- 「服务器数据恢复」重装系统导致XFS文件系统分区丢失的数据恢复
-
服务器数据恢复环境:DellPowerVault系列磁盘柜;用RAID卡创建的一组RAID5;分配一个LUN。服务器故障:在Linux系统层面对LUN进行分区,划分sdc1和sdc2两个分区。将sd...
- 服务器数据恢复-ESXi虚拟机被误删的数据恢复案例
-
服务器数据恢复环境:一台服务器安装的ESXi虚拟化系统,该虚拟化系统连接了多个LUN,其中一个LUN上运行了数台虚拟机,虚拟机安装WindowsServer操作系统。服务器故障&分析:管理员因误操作...
- 「服务器数据恢复」Raid5阵列两块硬盘亮黄灯掉线的数据恢复案例
-
服务器数据恢复环境:HPStorageWorks某型号存储;虚拟化平台为vmwareexsi;10块磁盘组成raid5(有1块热备盘)。服务器故障:raid5阵列中两块硬盘指示灯变黄掉线,无法读取...
- 服务器数据恢复—基于oracle数据库的SAP数据恢复案例
-
服务器存储数据恢复环境:某品牌服务器存储中有一组由6块SAS硬盘组建的RAID5阵列,其中有1块硬盘作为热备盘使用。上层划分若干lun,存放Oracle数据库数据。服务器存储故障&分析:该RAID5阵...
- 「服务器虚拟化数据恢复」Xen Server环境下数据库数据恢复案例
-
服务器虚拟化数据恢复环境:Dell某型号服务器;数块STAT硬盘通过raid卡组建的RAID10;XenServer服务器虚拟化系统;故障虚拟机操作系统:WindowsServer,部署Web服务...
- 服务器数据恢复—RAID故障导致oracle无法启动的数据恢复案例
-
服务器数据恢复环境:某品牌服务器中有一组由4块SAS磁盘做的RAID5磁盘阵列。该服务器操作系统为windowsserver,运行了一个单节点Oracle,数据存储为文件系统,无归档。该oracle...
- 服务器数据恢复—服务器磁盘阵列常见故障表现&解决方案
-
RAID(磁盘阵列)是一种将多块物理硬盘整合成一个虚拟存储的技术,raid模块相当于一个存储管理的中间层,上层接收并执行操作系统及文件系统的数据读写指令,下层管理数据在各个物理硬盘上的存储及读写。相对...
- 「服务器数据恢复」IBM某型号服务器RAID5磁盘阵列数据恢复案例
-
服务器数据恢复环境:IBM某型号服务器;5块SAS硬盘组成RAID5磁盘阵列;存储划分为1个LUN和3个分区:第一个分区存放windowsserver系统,第二个分区存放SQLServer数据库,...
- 服务器数据恢复—Zfs文件系统下误删除文件如何恢复数据?
-
服务器故障:一台zfs文件系统服务器,管理员误操作删除服务器上的数据。服务器数据恢复过程:1、将故障服务器所有磁盘编号后取出,硬件工程师检测所有硬盘后没有发现有磁盘存在硬件故障。以只读方式将全部磁盘做...
- 服务器数据恢复—Linux+raid5服务器数据恢复案例
-
服务器数据恢复环境:某品牌linux操作系统服务器,服务器中有4块SAS接口硬盘组建一组raid5阵列。服务器中存放的数据有数据库、办公文档、代码文件等。服务器故障&检测:服务器在运行过程中突然瘫痪,...
- 服务器数据恢复—Sql Server数据库数据恢复案例
-
服务器数据恢复环境:一台安装windowsserver操作系统的服务器。一组由8块硬盘组建的RAID5,划分LUN供这台服务器使用。在windows服务器内装有SqlServer数据库。存储空间LU...
- 服务器数据恢复—阿里云ECS网站服务器数据恢复案例
-
云服务器数据恢复环境:阿里云ECS网站服务器,linux操作系统+mysql数据库。云服务器故障:在执行数据库版本更新测试时,在生产库误执行了本来应该在测试库执行的sql脚本,导致生产库部分表被tru...
你 发表评论:
欢迎- 一周热门
-
-
爱折腾的特斯拉车主必看!手把手教你TESLAMATE的备份和恢复
-
如何在安装前及安装后修改黑群晖的Mac地址和Sn系列号
-
[常用工具] OpenCV_contrib库在windows下编译使用指南
-
WindowsServer2022|配置NTP服务器的命令
-
Ubuntu系统Daphne + Nginx + supervisor部署Django项目
-
WIN11 安装配置 linux 子系统 Ubuntu 图形界面 桌面系统
-
解决Linux终端中“-bash: nano: command not found”问题
-
NBA 2K25虚拟内存不足/爆内存/内存占用100% 一文速解
-
Linux 中的文件描述符是什么?(linux 打开文件表 文件描述符)
-
K3s禁用Service Load Balancer,解决获取浏览器IP不正确问题
-
- 最近发表
-
- 服务器数据恢复—Raid5数据灾难不用愁,Raid5数据恢复原理了解下
- 服务器数据恢复—多次异常断电导致服务器raid不可用的数据恢复
- 服务器数据恢复-V7000存储更换磁盘数据同步失败的数据恢复案例
- 「服务器数据恢复」重装系统导致XFS文件系统分区丢失的数据恢复
- 服务器数据恢复-ESXi虚拟机被误删的数据恢复案例
- 「服务器数据恢复」Raid5阵列两块硬盘亮黄灯掉线的数据恢复案例
- 服务器数据恢复—基于oracle数据库的SAP数据恢复案例
- 「服务器虚拟化数据恢复」Xen Server环境下数据库数据恢复案例
- 服务器数据恢复—RAID故障导致oracle无法启动的数据恢复案例
- 服务器数据恢复—服务器磁盘阵列常见故障表现&解决方案
- 标签列表
-
- linux 查询端口号 (58)
- docker映射容器目录到宿主机 (66)
- 杀端口 (60)
- yum更换阿里源 (62)
- internet explorer 增强的安全配置已启用 (65)
- linux自动挂载 (56)
- 禁用selinux (55)
- sysv-rc-conf (69)
- ubuntu防火墙状态查看 (64)
- windows server 2022激活密钥 (56)
- 无法与服务器建立安全连接是什么意思 (74)
- 443/80端口被占用怎么解决 (56)
- ping无法访问目标主机怎么解决 (58)
- fdatasync (59)
- 405 not allowed (56)
- 免备案虚拟主机zxhost (55)
- linux根据pid查看进程 (60)
- dhcp工具 (62)
- mysql 1045 (57)
- 宝塔远程工具 (56)
- ssh服务器拒绝了密码 请再试一次 (56)
- ubuntu卸载docker (56)
- linux查看nginx状态 (63)
- tomcat 乱码 (76)
- 2008r2激活序列号 (65)