1.1 Linux进程管理
Linux的进程管理与UNIX的进程管理相似。它包括进程调度、中断处理、信号、进程优先级、上下文切换、进程状态、进度内存等。
一个进程是处于执行期的程序(目标码存放在某种介质上)。但是不局限于一段可执行的代码,通常进程还包括其他资源,比如打开的文件,挂起的信号,内核内部数据,处理器状态,一个或多个具有内存映射的内存地址空间及一个或多个执行线程,还包括用来存储数据的数据段。
执行线程, 线程,是在进程中活动的对象。每个线程用于独立的的程序计数器,进程栈,和一组进程寄存器。内核调度的对象是线程而不是进程。linux系统线程的实现非常特别,对线程与进程并不特别区分。对Linux而言,线程是特殊的进程。
子进程通常运行他们自己的程序而不是与父进程运行相同的程序。这个操作避免了不必要的开销,因为复制整个地址空间是一个非常缓慢和效率低下的操作,它需要使用大量的处理器时间和资源。
当程序已经执行完成,子进程通过调用exit()系统调用终止。exit()系统调用释放进程大部分的数据并通过发送一个信号通知其父进程。此时,子进程是一个被叫作僵尸进程的进程。
子进程不会被完全移除直到其父进程知道其子进程的调用wait()系统调用而终止。当父进程被通知子进程终止,它移除子进程的所有数据结构并释放它的进程描述。
每一个进程拥有自己的状态,状态表示了进程当前在发生什么。在进程的执行期间进程的状态会发生改变。一些进程的状态如下:
TASK_RUNNING
在此状态下,表示进程正在CPU中运行或在队列中等待运行(运行队列)。
TASK_STOPPED
在此状态下的进程被某些信号(如SIGINT,SIGSTOP)暂停。进程正在等待通过一个信号恢复运行,例如SIGCONT。
TASK_INTERRUPTIBLE
在此状态下,进程被暂停并等待一个某些条件状态的到达。如果一个进程处于TASK_INTERRUPTIBLE状态并接收到一个停止的信号,进程的状态将会被改变并中断操作。一个典型的TASK_INTERRUPTIBLE状态的进程的例子是一个进程等待键盘中断。
TASK_UNINTERRUPTIBLE
与TASK_INTERRUPTIBLE相似。当一个进程处于TASK_UNINTERRUPTIBLE状态可以被中断,向处于TASK_UNINTERRUPTIBLE状态的进程发送一个信号不会发生任何操作。一个TASK_UNINTERRUPTIBLE进程的典型的例子是等待磁盘I/O操作。
TASK_ZOMBIE
当一个进程调用exit()系统调用退出后,它的父进程应该知道该进程的终止。处于TASK_ZOMBIE状态的进程会等待其父进程通知其释放所有的数据结构
进程内存区由以下几部分组成:
Text段
该区域用于存储运行代码。
Data段
数据段包括三个区域。
- Data:该区域存储已被初始化的数据,如静态变量。
- BSS:该区域存储初始化为0的数据。数据被初始化为0。
- Heap:该区域用于根据需求使用malloc()动态申请的内存。堆向高地址方向增长。
Stack段
该区域用于存储局部变量、函数参数和返回函数的地址。栈向低地址方向增长。
用户进程的地址空间内存分布可以使用pmap命令来查看。你可以使用ps命令来查看内存段的大小