欢迎关注我的博客Zhuhao's Blog获取最新文章
简单的说,每个应用在运行时就会产生一个进程,这个进程就对这个应用负责,掌握这个应用的运行状态。
可是为什么还要用一个进程来控制一个应用呢,下面将会简单的解释一下。
现在的应用对于资源的要求都是狮子大开口,开口就是几个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_SUCCESS和 EXIT_FAILURE 这两个可移植的宏分别表示成功和失败(也可以使用0和非0值来表示,不过可移植性就差了)。
简单的表示成功退出就使用
exit(EXIT_SUCCESS);
这个系统调用会先完成在用户空间需要做的事,再调用_exit()再处理内核中的事。
在用户空间做的事
- 调用任何由 atexit() 或 on_exit() 注册的函数,按在系统中注册的相反顺序。(假如在exit()前运行了atexit(a) atexit(b),那么在调用exit()后就会先运行b再运行a )
- 清空所有已打开的标准I/O流
- 删除有tmpfile()函数创建的所有临时文件
内核会清理进程所创建的 不再使用的所有资源这包括但不局限于:分配内存 打开文件和System Ⅴ的信号量。清理完成后,内核会摧毁进程,并告知父进程其子进程已经终止。