概述
进程的定义
程序:
程序是存放在存储介质上的一个可执行文件。
进程:
进程是程序的执行实例,包括程序计数器,寄存器和变量的当前值。程序是静态的,进程是动态的。
程序是一些指令的有序集合,而进程是程序执行的过程。进程是程序的一次执行过程。进程的状态是变化的,其包括进程的创建,调度和消亡。
只要程序运行,此时就是进程,程序每运行一次,就会创建一个进程。
在linux系统中,进程是管理事务的基本单元。
进程拥有自己独立的处理环境和系统资源(处理器,存储器,I/O设备,数据,程序)。
可用exec函数由内核将程序读入内存,使其执行起来称为一个进程。
进程的状态及转换
进程整个生命周期可以简单划分为三种状态:
就绪态:
进程已经具备执行的一切条件,正在等待分配cpu的处理时间。
执行态:
该进程正在占用CPU运行。
等待态:
进程因不具备某些执行条件而暂时无法继续执行的状态。
进程的调度机制:
时间片轮转,上下文切换
多进程不是说一个进程执行完再执行另一个进程,而是交替执行的,一个进程执行一段时间,然后下一个进程再执行一段时间,以此类推,所有进程执行完之后再回到第一个进程继续执行,以此类推。
进程控制块
进程控制块就是用于保存一个进程信息的结构体,又称之为PCB
OS是根据PCB来对并发执行的进程进行控制和管理的,系统在创建一个进程的时候会开辟一段内存空间存放与此进程相关的PCB数据结构。
PCB是操作系统中最重要的记录型数据结构。PCB中纪律了用于描述进程进展情况及控制进程运行所需的全部信息。
PCB是进程存在的唯一标志,在linux中PCB存放在task——struct结构体中。
调度数据:
进程的状态,标志,优先级,调度策略等。
PCB结构体中的部分数据
时间数据:
创建该进程的时间,在用户态的运行时间,在内核态的运行时间等。
文件系统数据:
umask掩码,文件描述符表等。内存数据,进程上下文,进程标识(进程号)
进程控制
进程号
每个进程都由一个进程号来标识,其类型为pid_t,进程号的范围:0~32767
进程号是由操作系统随机给当前进程分配的,不能自己控制
进程号总是唯一的,但进程号可以重用,当一个进程终止后,其进程号就可以再次使用了。
在ubuntu中查看当前系统中所有的开启的进程
ps ajx
PPID:当前进程的父进程的进程号,任何进程(除init进程)都是由另一个进程创建,该进程称为被创建进程的父进程。
PID:当前进程的进程号,标识进程的一个非负整型数。
PGID:当前进程所在的组的进程组ID,进程组是一个或多个进程的集合。他们之间相互关联,进程组可以接收同一终端的各种信号,关联的进程有一个进程组号。
COMMAND:当前进程的名字
特殊的进程号:
在linux系统中进程号由0开始。
进程号为0及1的进程由内核创建。
进程号为0的进程通常是调度进程,常被称为交换进程(swapper)。
进程号为1的进程通常是init进程,init进程是所有进程的祖先。
除调度进程外,在linux下面所有的进程都由进程init进程直接或间接创建。
linux操作系统提供了三个获得进程号的函数getpid(),getppid(),getpgid()。
#include<sys/types.h>
#include<unistd.h>
pid_t getpid(void);
功能:获取当前进程的进程号
pid_t getppid(void);
功能:获取当前进程的父进程的进程号
pid_t getpgid(void);
功能:获取当前进程所在进程组的id
进程的创建fork函数
在linux环境下,创建进程的主要方法时调用以下两个函数:
#include<sys/types.h>
#include<unistd.h>
pid_t fork(void);
pid_t vfork(void);
创建一个新进程:
pid_t fork(void);
功能:
fork()函数用于一个已存在的进程中创建一个新进程,新进程称为子进程,原进程称为父进程。
返回值:
成功:子进程返回0,父进程中返回子进程的id。
失败:返回-1。
使用fork函数得到的子进程是父进程的一个复制品,它从父进程处继承了整个进程的地址空间。
地址空间:
包括进程上下文,进程堆栈,打开的文件描述符,信号控制设定,进程优先级,进程组号。
fork函数执行完毕后父子进程的空间示意图:
创建子进程
不区分父子进程(不推荐)
区分父子进程
父子进程拥有独立的地址空间:
子进程继承父进程的空间:
进程的挂起
进程在一定的时间内没有任何动作,称为进程的挂起。
#include<unistd.h>
unsigned int sleep(unsigned int sec);
参数:
seconds:指定要挂起的秒数
功能:
进程挂起指定的秒数,直到指定的时间用完或收到信号才解除挂起。
返回值:
若进程挂起到sec指定的时间则返回0,若有信号中断则返回剩余秒数。
注意:
进程挂起指定得秒数后程序并不会立即执行,系统只是将此进程切换到就绪态。
## 进程的等待
父子进程有时需要简单的进程同步,如父进程等待子进程的结束
linux下提供了以下两个等待函数wait(),waitpid()。
需要包含头文件:
#include<sys/types.h>
#include<sys/wait.h>
wait函数
pid_t wait(int *status);
功能:等待子进程终止,如果子进程终止了,此函数会回收子进程的资源。
调用wait函数的进程会挂起,直到它的下一个子进程退出或者收到一个不能被忽视的信号时才被唤醒。
若调用进程没有子进程或它的子进程已经结束,该函数立即返回。
参数:
函数返回时,参数status中包含子进程退出时的状态信息,子进程的退出信息在一个int中包含了多个字段,用宏定义可以取出其中的每个字段。
返回值:
如果执行成功则返回子进程的进程号。
出错返回-1,失败原因存于errno中。
取出子进程的退出信息
WIFEXITED(status)
如果子进程是正常终止的,取出的字段值非零。
WEXITSTATUS(status)
返回子进程的退出状态,退出状态保存在status变量的8~16位。在用此宏前应先用宏WIFEXITED判断子进程是否正常退出,正常退出才可以使用此宏。
注意:
此status是个wait的参数指向的整型变量。
waitpid函数
pid_t waitpid(pid_t pid,int *status,int options)
功能:
等待子进程终止,如果子进程终止了,此函数会回收子进程的资源。
返回值:
如果执行成功返回子进程的ID。
出错返回-1,失败原因存于errno中。
参数pid的值有以下几种类型:
pid>0:
等待 进程ID等于pid的 子进程。
pid=0:
等待同一个进程组中的任何子进程,如果子进程已经加入了别的进程组,waitpid不会等待它。
pid=-1:
等待任一子进程,此时waitpid和wait作用一样。
pid<-1:
等待指定进程组中的任何子进程,这个进程组的ID等于pid的绝对值。
status参数中包含子进程退出时的状态信息。
options参数能进一步控制waitpid的操作:
0:
同wait,阻塞父进程,等待子进程退出。
WNOHANG:
没有任何已经结束的子进程,则立即返回。
WUNTRACED:
如果子进程暂停了则此函数马上返回,并且不予以理会子进程的退出状态。(跟踪调试,很少用到)
返回值:
成功:
返回状态改变了的子进程的进程号,如果设置了选项WNOHANG并且pid指定的进程存在则返回0。
出错:
返回-1,当pid所指示的子进程不存在,或此进程存在,但不是调用进程的子进程,waitpid就会出错返回,这时errno被设置为ECHILD。
wait(status)<==>waitpid(-1, status, 0)
案例:
特殊进程:
僵尸进程:
进程已运行结束,但进程的占用的资源未被回收,这样的进程称为僵尸进程。
子进程已运行结束,父进程未调用wait或waitpid函数回收子进程的资源是子进程变为僵尸进程的原因。
孤儿进程:
父进程运行结束,但子进程为运行结束的子进程。
守护进程:
守护进程是个特殊的孤儿进程,这种进程脱离终端,在后台运行。
进程的终止
在linux下可以通过以下方式结束正在运行的进程:
void exit(int value);
void _exit(int value);
exit函数
结束进程执行
#include<stdlib.h>
void exit(int value)
参数:
status:返回给父进程的参数(低8位有效)。
_exit函数
#include<unistd.h>
void _exit(int value)
参数:
status:返回给父进程的参数(低8位有效)。
#include<unistd.h>
void _exit(int status);
功能:退出当前进程
参数:
status:退出状态,由父进程通过wait函数接收这个状态
一般失败退出设置为非0
一般成功退出设置为0
返回值:
无
#include<unistd.h>
void exit(int status);
功能:退出当前进程
参数:
status:退出状态,由父进程通过wait函数接收这个状态
一般失败退出设置为非0
一般成功退出设置为0
exit和_exit函数的区别:
exit为库函数,而_exit为系统调用
exit会刷新缓冲区,但是_exit不会刷新缓冲区
一般会使用exit
![在这里插入图片描述](https://img-blog.csdnimg.cn/20201125110911387.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQ1NzQ1ODQz,size_16,color_FFFFFF,t_70#pic_center
进程退出清理
进程在退出前可以用atexit函数注册退出清理函数
#include<stdlib.h>
int atexit(void(*function)(void));
功能:
注册进程正常结束前调用的函数,进程退出执行注册函数。
参数:
function:进程结束前,调用函数的入口地址。
一个进程中可以多次调用atexit函数注册清理函数,正常结束前调用函数的顺序和注册时的顺序相反。
进程的创建——vfork函数
pid_t vfork(void)
功能:
vfork函数和fork函数一样都是在已有的进程中穿件一个新的进程,但它们创建的子进程是有区别的。
返回值:
创建子进程成功,则在子进程中返回0,父进程中返回子进程的ID。出错则返回-1。
fork和vfork函数的区别:
vfork保证子进程先运行,在它调用exec或exit之后,父进程才可能被调度运行。
vfork和fork一样都创建一个子进程,但他并不将父进程的地址完全复制到子进程中,因为子进程会立即调用exec或(exit),于是也就不访问改地址空间。
相反,在子进程中调用exec或exit之前,他在父进程的地址空间中运行,在exec之后子进程会有自己的进程空间。
进程的创建——vfork函数
#include<sys/types.h>
#include<unistd.h>
pid_t vfork(void);
## 子进程在父进程之前运行
## 子进程和父进程共享同一块空间
进程的替换
exec函数族,是由六个exec函数组成的。
1 exec函数族提供了六种在进程中启动另一个程序的方法。
2 exec函数族可以根据指定的文件名或目录名找到可执行文件。
3 调用exec函数的进程并不创建新的进程,故调用exec前后,进程的进程号并不会改变,骑执行的程序完全由新的程序替换,而新程序则从其main函数开始执行。
exec函数族取代调用进程的数据段,代码段和堆栈段。
exec函数族
#include<unistd.h>
int execl(const char *pathname,
const char *arg0,…,
NULL);
int execlp(const char *filename,
const char *arg0,…,
NULL);
int execle(const char *pathname,
const char *arg0,…,NULL,
char *const envp[]);
int execv(const char *pathname,
const char *argv[]);
int execvp(const char *filename,
char *const argv[]);
int execve(const char *pathname,
char *const argv[],
char *const envp[]);
六个exec函数中只有execve是真正意义的系统调用(内核提供的接口),其它函数都是在此基础上经过封装的库函数。
l(list):
参数地址列表,以空指针结尾。
参数地址列表
char *arg0, char *arg1,…, char *argn, NULL
v(vector)
存有各参数地址的指针数组的地址
使用时先构造一个指针数组,指针数组存各参数的地址,然后将该指针数组地址作为函数的参数。
p(path)
按PATH环境变量指定的目录搜素可执行文件。
以p结尾的exec函数取文件名作为参数,当指定filename作为参数时,若filename中包含/,则将其视为路径名,并直接到指定的路径中执行程序。
e(environment)
存在环境变量字符串地址的指针数组的地址,execle和execve改变的是exec启动的程序的环境变量(新的环境变量完全由environment指定),其他四个函数启动的程序则使用默认系统的环境变量。
注意:
exec函数与一般的函数不同,exec函数族中的函数执行成功后不会返回,只有调用失败了,他们才会返回-1,失败后从原程序的调用点接着往下执行。
在平时的编程中,如果用到了exec函数族,一定要记得加错误判断语句。
一个进程调用exec后,除了进程ID,进程还保留了下列特征不变
父进程号
进程组号
控制终端
根目录
当前工作目录
进程信号屏蔽集
未处理信号
。。。。。
最后
以上就是专一水蜜桃为你收集整理的linux高级程序设计——进程的全部内容,希望文章能够帮你解决linux高级程序设计——进程所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
发表评论 取消回复