1.进程组
在Linux下,每一个进程都有自己的一个进程ID,但是他还包含了一个进程组ID,它是属于一个进程组的。那么进程组是什么呢?
进程组是由一个或多个的进程组成的集合。每个进程组有一个唯一的进程组ID,也可以说是这个进程组的编号。那么既然是一个组,就要有一个组长进程,既然是组长进程,那么该进程组的组ID就等于组长进程的ID。
组长进程可以创建一个进程组,并且创建该组中的进程,同时也拥有权力去终止创建的进程。
在一个进程组中,如果组长进程不存在,但是该组中还有别的进程存在,那么该进程组就一直存在,与组长进程是否存在没有关系。
创建了三个进程

可以看到他们的PGID是一样的,是第一个进程(也就是组长进程)的PID
杀死组长进程后,该组仍旧存在
2.作业
Shell分前后台来进行控制的不是进程而是作业或者是进程组。
一个前台作业和一个后台作业都可以由多个进程组成。
作业控制 : 在Shell里,可以运行一个前台作业和多个后台作业
作业与进程组的区别:如果作业中的某个进程又创建了子进程,子进程不属于作业,但是属于进程组
如果某个作业一旦结束,Shell就会把自己提到前台,但是原来的进程组创建的子进程还在,可是子进程并不属于作业,这个时候 如果原来的前台进程没有运行完毕(其实这里说的是子进程)的话,会自动变为后台进程组(前提是该子进程得存在没有被终止)
或者可以这样说,当前台有某个作业运行的时候,因为前台只能运行一个作业,这个时候Shell就会自动被运行到后台。当这个前台作业运行完毕后,Shell就会被提到前台,可以继续接受用户的输入
创建一个进程然后fork出一个子进程,i为0的时候,父进程会退出,但是子进程还在运行,但是此时的前台作业因为父进程的退出变成了Shell,子进程被自动提到了后台,此时可以显示Shell了
但是父进程也就是组长进程虽然退出了,但是子进程所属的组还在,然后杀掉即可
3.会话
会话的概念 : 会话是一个或者多个进程组的集合。
一个会话可以有一个控制终端。
控制进程:建立与控制终端连接的会话首进程
会话包含的内容 : 会话首进程(控制进程)、一个前台进程组和一个或多个后台进程组
在这里我们创建了三个进程,它们属于同一个进程组,而且它们的SID是一样的,SID是会话id,所以属于同一个会话,而这个会话ID1发现是bash(解释器),也就是说bash是会话首进程,并且是这三个进程的的父进程。
4.作业控制
在Shell里面,分前后台进行控制的不是进程而是作业或者是进程组
一个前台作业可以由多个进程组组成,一个后台作业也可以由多个进程组组成
作业控制:在Shell里面,可以同时运行一个前台作业和任意多个后台作业便称之为作业控制
新建两个作业至后台
jobs可以查看当前有哪些作业
fg将作业1提至前台运行,然后ctrl+z可以将其提至后台,将其提至后台那么它在前台的运行就停止了,这个时候jobs可以看到前台作业1已经stop了
bg命令可以使某个停止的作业继续在后台运行,那么在前台终止的作业1提至后台后,用bg就又处于运行状态了
注意:前台作业不能被stop
5.守护进程
守护进程(精灵进程),它是一种运行在后台的特殊进程,它独立于终端并且7*24小时的运行
守护进程是一种很有用的进程。在Linux上大多数的服务器就是用守护进程来实现的,例如有:ssh服务器、Web服务器HTTP,FTP服务器等等,并且守护进程还能完成许多系统任务,如可以规划进程crond。
在Linux下,当我们进行启动时,同时也会有许多的系统服务进程也进行启动,不能直接和用户进行交互。
别的进程都是在用户登录或者程序在运行的时候进行创建,在程序运行结束的时候或者用户注销时终止。
守护进程并不受用户登录或者注销的影响,它们一直在运行着
>>>守护进程的标志 : 凡是在TPGID一栏写着-1的都是没有控制终端的进程(守护进程)
守护进程通常采用以d结尾的名字,表示Daemon
在COMMEND一列用【】括起来的名字表示内核线程,通常采 用 Kernel表示
>>> 创建守护进程 : 最关键的一步是调用setid函数创建一个新的Session,并成为Session Leader。
接口 :
#include <unsitd.h>
pid_t stdsid(void); //函数调用成功的时候返回新创建的进程id,出错返回-1
注意 : 在调用这个函数之前,当前进程是不允许是进程组的组长,否则函数会返回-1
要保证当前进程不是进程组的组长,可以调用fork在调用sedsid。
1.#include <stdio.h>
2 #include <signal.h>
3 #include <unistd.h>
4 #include <stdlib.h>
5 #include <sys/stat.h>
6 #include <fcntl.h>
7
8 void mydaemon( void )
9 {
10 int i = 0;
11 pid_t pid;
12 int fd0;
13 struct sigaction sa;
14
15 umask(0); // 调用umask将文件模式创建屏蔽子设置为0
16
17 //2.调用fork,使得父进程推出
18
19 //如果守护进程是一条简单的shell命令启动的,当父进程终止后,那么shell就认为
20 //该命令已经执行完毕
21
22 //保证子进程不是一个进程组的组长进程
23 if( (pid = fork()) < 0 ){
perror("fork()");
25 }
26 else if( pid > 0 ){
27 exit(0); // 父进程终止
28 }
29
30 sa.sa_handler = SIG_IGN; // 忽略SIGCHLD信号
31 setsid();//创建一个新的会话(子进程)
32 if(sigaction(SIGCHID,&sa,NULL) < 0)
33 return;
34
35
36 close(0);
37 fd0 = open("/dev/null",O_RDWR);
38 dup2(fd0,1);
39 dup2(fd0,2);
结果:
