简单的说,每个应用在运行时就会产生一个进程,这个进程就对这个应用负责,掌握这个应用的运行状态。

可是为什么还要用一个进程来控制一个应用呢,下面将会简单的解释一下。

现在的应用对于资源的要求都是狮子大开口,开口就是几个G,一台电脑的内存一般也就几个G,总不能 一台电脑就跑这一个应用吧。为了解决这个问题,操作系统就使用了虚拟内存,让每个进程代表一个 应用,给每个进程一种自己独霸整台电脑的假象,然后操作系统进行上下文切换,只把这个进程正好 需要使用的资源放进内存;这样每个进程都有自己独有的资源。

创建进程

创建进程需要一个系统调用 fork(),fork可以创建一个和当前进程映像一样的进程;成功时 创建子进程并返回子进程的pid,失败时不会创建子进程,返回-1并设置相应的errno。

顺便介绍一下pid,pid是进程的ID,数据类型是pid_t,在Linux中被定义为int。可以调用 getpid() 来获得调用进程的pid,还可以通过调用 getppid() 来获得调用进程的父进程的pid。

运行进程

运行进程需要调用 exec 系统调用,但是不存在单一的exec函数,他是由一系列的exec函数组成的。

以一个最简单的调用 execl() 为例: + 成功的execl()调用改变 地址空间 和进程映像 + 所有的挂起的信号都会丢失 + 捕捉到的所有信号都会还原为默认处理方式 + 丢弃所有的内存锁 + 大多数进程的属性会还原成默认值(pid 父进程的pid 优先级 所属的用户和组 不会变) + 清空和进程内存地址空间相关的所有数据,包括所有映射的文件

终止进程

终止进程使用的是 exit() 系统调用,参数用于表示进程的推出状态, EXIT_SUCCESSEXIT_FAILURE 这两个可移植的宏分别表示成功和失败(也可以使用0和非0值来表示,不过 可移植性就差了)。

简单的表示成功退出就使用

1
exit(EXIT_SUCCESS);

这个系统调用会先完成在用户空间需要做的事,再调用_exit()再处理内核中的事。

在用户空间做的事 + 调用任何由 atexit()on_exit() 注册的函数,按在系统中注册的相反顺序。( 假如在exit()前运行了atexit(a) atexit(b),那么在调用exit()后就会先运行b再运行a ) + 清空所有已打开的标准I/O流 + 删除有tmpfile()函数创建的所有临时文件

内核会清理进程所创建的 不再使用的所有资源这包括但不局限于:分配内存 打开文件和System Ⅴ 的信号量。清理完成后,内核会摧毁进程,并告知父进程其子进程已经终止。

参考源