Unix操作系统调度的基本单位是进程,单个或多个进程组成进程组,单个或多个进程组则组成会话。其关系示意图如下:
进程组
#include <unistd.h>
pid_t getpgrp(); // 返回调用进程的进程组ID
pid_t getpgid(pid_t pid); // 返回指定进程的进程组ID,若失败返回-1
int setpgid(pid_t pid, pid_t pgid); // 设置指定进程的进程组ID,成功返回0,失败返回-1一个进程组中有一个组长进程和0个或多个组员进程,组长进程的进程ID等于其进程组ID。所以,可以通过调用setpgid(getpid())创建一个进程组。
会话
#include <unistd.h>
pid_t setsid(); // 若成功则返回进程组ID,出错返回-1如果调用此函数的进程不是组长进程,那么此函数将创建一个新会话。调用此函数将发生:
- 该进程成为新会话的首进程,也是新会话中的唯一进程
- 该进程成为新进程的组长进程
- 若该进程有控制终端,则会与控制终端断开联系
控制终端
会话和进程组与控制终端的联系:
- 一个会话可以有一个控制终端
- 建立与控制终端连接的会话首进程被称为控制进程
- 如果一个会话有一个控制终端,则它有一个前台进程组,会话中的其他进程组则称为后台进程组
- 中断(SIGINT)和退出(SIGQUIT)信号只会发给会话中的前台进程组
函数
#include <unistd.h>
pid_t tcgetpgrp(int filedes); // 成功则返回前台进程组的进程组ID,出错返回-1
int tcsetpgrp(int filedes, pid_t pgid); // 设置前台进程组,成功返回0,失败返回-1
系统启动示意图
Shell命令执行过程
以命令ps -o pid,ppid,pgid,sid,tpgid,comm | cat tmp命令为例: