进程组
1.概念
进程组是一个或者多个进程的集合
2.特性
<1>.每个进程组有唯一的进程组ID
<2>.每个进程组有一个组长进程,组长进程的标识是,其进程组ID等于其进程ID
<3>.只要在进程组中有一个进程存在,该进程组就存在,这与其组长进程是否终止无关
3.命令ps
<1>.功能:查看进程信息
<2>.选项:
a:不仅列当前用户的进程,也列出所有其他用户的进程
x:表示不仅列有控制终端的进程,也列出所有无控制终端的进程
j:表示列出与作业控制相关的信息
注意:
进程组一般放到后台运行(&),因为放到前台,ctrl+c后,所有的进程都会被杀
作业(Job)
特点
<1>.Shell分前后台来控制的不是进程而是作业(Job)或者进程组
<2>.Shell可以运行一个前台作业和任意多个后台作业,这称为作业控制
<3>.bash默认是一个前台作业
作业与进程组的区别:
当作业中的某个进程创建子进程,则子进程不属于作业,但却属于进程组
会话(Session)
1.概念
会话是一个或者多个进程组的集合
2.特点
<1>.一个会话可以有一个控制终端
<2>.一个会话中的几个进程组可被分为一个前台进程组及一个或多个后台进程组
<3>.一个会话中,应该包括控制进程(会话的首进程,默认是bash),一个前台进程组和任意个后台进程组(默认的前台进程组是bash)
<4>.SID:会话id,每打开一个终端,就新建了一个会话
守护进程
特点
1.守护进程独立于控制终端(与控制终端无关)
2.大部分的守护进程是孤儿进程,自成进程组、自成会话
3.周期性的执行某种任务或等待处理某些发生的事件
4.完成许多系统任务,如:crond定时任务
5.守护进程不受用户注销的影响,他们一直在运行着
6.守护进程通常采用以d结尾的名字,表示Deamon
创建守护进程
调用setsid函数创建一个新的Session,并成为Session Leader
成功返回新创建的Session的id,出错返回-1
#include <unistd.h>
void setsid(voi)
注意:调用setsid函数之前,当前进程不允许是进程组的Leader,否则该函数返回-1
解决方法:先fock,在调用setsid
简单实现:
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <signal.h>
void mydeamon(void){
int fd0;
pid_t pid;
struct sigaction sa;
umask(0);//1.将文件模式创建屏蔽字设置为0
//2.调用fork,父进程退出
if((pid = fork()) < 0){
perror("fork error");
}else if(pid > 0){//父进程
exit(0);
}
//3.调用setsid创建一个会话
setsid();
sa.sa_handler = SIG_IGN;//4.忽略SIGCHLD信号
sigemptyset(&sa.sa_mask);
sa.sa_flags = 0;
if(sigaction(SIGCHLD,&sa,NULL) < 0){
return;
}
//再次forc终止父进程,保持子进程不是话首进程,从而保证后续不会在和其他终端关联
//不是必须
if((pid = fork()) < 0){
perror("fork");
}else if(pid > 0){
exit(0);
}
//5.将当前目录更改为跟目录
if(chdir("/") < 0){
printf("child dir error");
return;
}
//关闭不在需要的文件描述符,或者重定向到 /dev/null直接丢弃
close(0);
fd0 = open("/dev/null",O_RDWR);
dup2(fd0,1);
dup2(fd0,2);
}
int main(){
mydeamon();
while(1){
sleep(1);
}
return 0;
}