【Linux网络】深入理解守护进程(Daemon)及其实现原理
进程组什么是进程组我们都知道进程拥有一个进程ID(PID。此外我们还会发现另一个IDPGID这个代表就是进程组ID。代码语言javascriptAI代码解释hychyc-alicloud:~/linux/Test$ ps -ajx | head -1 ps -ajx |grep test PPID PID PGID SID TTY TPGID STAT UID TIME COMMAND 46661 46817 46817 46661 pts/0 46821 S 1001 0:00 ./test一个进程必然属于一个进程组。一个进程组中可以有一个 or 多个进程。进程组组长一个进程组中存在一个组长进程。当一个进程的PID PGID时那么这个进程就是它对应进程组的组长。代码语言javascriptAI代码解释hychyc-alicloud:~/linux/Test$ ps -ajx | head -1 ps -ajx |grep test PPID PID PGID SID TTY TPGID STAT UID TIME COMMAND 46661 46817 46817 46661 pts/0 46821 S 1001 0:00 ./test进程组组长的作用组长可以创建一个进程组 or 在进程组中创建进程进程组的生命周期从进程组被创建开始到进程组中最后一个进程被kill为止。注进程组的生命周期与组长的生命周期无关只要进程组中还有进程进程组就依然存在通过管道执行多条命令我们可以发现它们都是属于同一个“进程组”。代码语言javascriptAI代码解释hychyc-alicloud:~$ sleep 100 | sleep 200 | sleep 300 [1] 2517 hychyc-alicloud:~$ ps -ajx | grep sleep PPID PID PGID SID TTY TPGID STAT UID TIME COMMAND 2499 2515 2515 2499 pts/0 2536 S 1001 0:00 sleep 100 2499 2516 2515 2499 pts/0 2536 S 1001 0:00 sleep 200 2499 2517 2515 2499 pts/0 2536 S 1001 0:00 sleep 300 2499 2537 2536 2499 pts/0 2536 S 1001 0:00 grep --colorauto sleep由此可见进程都是由进程组的形式完成对应的任务的若一次只启动了一个进程那么就单个进程形成一个进程组。会话什么是会话上面我们谈到了进程组会话其实与进程组紧密相关会话是一个 or 多个进程组的集合一个会话可以包含一个 or 多个进程组通常我们使用管道将几个进程编成一个进程组。如上图中的进程组2、进程组3。代码语言javascriptAI代码解释[nodelocalhost code]$ proc2 | proc3 [nodelocalhost code]$ proc4 | proc5 | proc6 # 表⽰将进程组放在后台执⾏会话中第一个进程组我们叫做会话首进程。会话首进程是会话的管理者会话ID(SID)会话ID既表示当前会话的ID。如下我们看到SID都是一样的这说明当下说有进程(进程组)都是在同一个会话中代码语言javascriptAI代码解释PPID PID PGID SID TTY TPGID STAT UID TIME COMMAND 2499 2515 2515 2499 pts/0 2536 S 1001 0:00 sleep 100 2499 2516 2515 2499 pts/0 2536 S 1001 0:00 sleep 200 2499 2517 2515 2499 pts/0 2536 S 1001 0:00 sleep 300 2499 2537 2536 2499 pts/0 2536 S 1001 0:00 grep --colorauto sleep在一般情况下一个用户执行的所有进程(进程组)都是在同一个会话中的因为当用户登录时系统会自动为用户创建一个会话。而在这个新建的会话中会有一个默认是会话首进程bash既进程组1与此同时系统还有创建一个终端文件(/dev/pts/xxx)并将会话进程组bash的标准输入、输出、错误重定向到终端文件并且将bash的PID设置为会话的SID(bash成为会话首进程)由此完成用户的登录此时的会话首进程又叫做前台进程(后面讲)。如何创建会话可以调用函数setsid()来创建新会话但前提是调用该函数的进程不能是进程组组长代码语言javascriptAI代码解释#include unistd.h /* *功能创建会话 *返回值创建成功返回SID, 失败返回-1 */ pid_t setsid(void);调用setsid()效果创建一个新会话将调用该函数的进程设置为新会话的会话首进程 and 设置为当前进程组的组长特殊机制如果调用该函数的进程原本有一个控制终端这个联系会被切断新会话默认是没用任何控制终端的 注意如果调用该函数的进程是进程组组长那么会报错所以为了避免这种情况我们通常是将调用fork()创建子进程然后再让父进程退出、子进程调用setsid()创建新会话 即使父进程是进程组组长也没用任何影响因为进程组的生命周期不由组长是否存在决定只要进程组中还有进程进程组就一直存在控制终端什么是控制终端控制终端 (Controlling Terminal)是一个与 会话 (Session) 绑定的终端设备。绑定关系它由会话首进程通常是登录 Shell打开并绑定整个会话内的所有进程默认共享该终端。因为控制终端的信息是保存在PCB中的所以当Shell进程fork创建子进程子进程的控制终端也是这个终端。核心特权它不仅仅是输入输出设备更具备作业控制能力。它能根据键盘输入向前台进程组发送硬件中断信号如 SIGINT, SIGTSTP并在断开连接时向会话首进程发送挂断信号SIGHUP。唯一性一个会话最多只能有一个控制终端可以没有。在默认没用重定向的情况下每一个进程的标准输入、输出、错误都是指向这个控制终端的会话、进程组以及控制终端的关系建立与控制终端连接的会话叫做控制进程如何一个会话拥有一个控制终端则它有一个前台进程组、以及多个后台进程组前台与后台进程在一个会话中最多有一个前台进程组、以及多个后台进程组。当什么都不做时Shell进程是默认在前台的而当我们启动任务时默认是作为前台任务执行此时Shell进程进入睡眠到后台等待我们执行的任务占据前台。当执行的任务完成后内核唤醒Shell进程并主动切换为前台进程。可以在指令最后加上‘’显式的让其在后台运行。介绍相关指令代码语言javascriptAI代码解释jobs查看系统当前的后台进程 ctrl c终止前台进程、对后台进程没有作用 ctrl z暂定前台进程自动切换为后台进程 fg %%把最近的后台作业重新拉回到前台 bg 任务号让后台进程运行起来作业控制作业与作业控制作业是针对用户来讲的其本质就是用户为了完成某项任务而启动的进程(组)。一个作业既可以包含一个进程也可以包含多个进程(进程之间相互合作完成任务通常使用管道)作业控制Shell可以分前后台来控制一个前台作业与多个后台作业。Shell同时运行一个前台作业与多个后台作业就被称为作业控制作业号当后台进程执行完后会返回一个作业号以及一个进程号(PID)或者通过jobs查看后台进程也可以查看到作业号以及进程号(PID)代码语言javascriptAI代码解释hychyc-alicloud:~$ sleep 1 [1] 3447 hychyc-alicloud:~$ jobs [1]- Running sleep 100 [2] Running sleep 200 当后台进程有多个时我们会发现作业号后面带有正负号。其中 表示默认作业- 表示即将称为默认作业。默认作业 ()你最后一次放手不管的那个任务最近操作过的。跟数字大小无关通常是最大的最新的那个但如果你回头去操作了老的任务老的也会变成默认的。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2582370.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!