守护进程
在linux系统中,我们会发现在系统启动的时候有很多的进程就已经开始跑了,也称为服务,这也是我们所说的守护进程。
守护进程是脱离于终端并且在后台运行的进程,脱离终端是为了避免在执行的过程中的信息在终端上显示,并且进程也不会被任何终端所产生的终端信息所打断。
守护进程一般的生命周期是系统启动到系统停止运行,当人,什么都敌不多人,也可以通过杀死进程的方式来结束进程的生命周期。linux系统中有很多的守护进程,最典型的就是我们经常看到的服务进程。当然,我们也经常会利用守护进程来完成很多的系统或者自动化任务。
创建守护进程的步骤:
(1)创建子进程,父进程退出
因为守护进程是脱离终端控制的,所以要造成一种在终端里已经运行完的假象,把所有的工作都放在子进程中去完成。我们知道,父进程退出后,子进程其实就是变成了孤儿进程,孤儿进程一般是由1号进程收养,也就是我们所谓的init进程,也就是说原来的子进程变成了init的子进程。
(2)创建会话
首先看一下进程组的概念:
进程组 :
1.每个进程也属于一个进程组
2.每个进程主都有一个进程组号,该号等于该进程组组长的PID号 .
3.一个进程只能为它自己或子进程设置进程组ID号
会话期:会话期(session)是一个或多个进程组的集合。
setsid()函数可以建立一个对话期:
如果,调用setsid的进程不是一个进程组的组长,此函数创建一个新的会话期。
(1)此进程变成该对话期的首进程
(2)此进程变成一个新进程组的组长进程。
(3)此进程没有控制终端,如果在调用setsid前,该进程有控制终端,那么与该终端的联系被解除。 如果该进程是一个进程组的组长,此函数返回错误。
(4)为了保证这一点,我们先调用fork()然后exit(),此时只有子进程在运行
其实就是起到了进程脱离原会话的控制,摆脱了原进程组的控制,摆脱了原控制终端的控制。
那么为什么要调用setsid函数呢?主要是因为前面创建子进程的时候,随人父进程退出了,但是子进程拷贝了父进程的会话、进程组、控制终端等等,所以需要使用setsid来摆脱控制。
(3)改变当前目录为根目录
在fork子进程的时候,子进程也继承了父进程的工作目录。通常是让‘/’目录作为守护进程的当前目录,可以避免很多麻烦。
(4)设置文件掩码
同一个文件的权限掩码是一样的,我们使用fork创建的子进程会从父进程那里继承一些已经打开的文件,打开的文件可能永远不会被守护进程读写,但是他们一样消耗资源,而且可能导致所在的文件系统无法结束。
守护进程实例:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#define MAXFILE 65535
int main()
{
int i,fd;
pid_t pid;
char *buf = "hello,dameon";
int len = strlen(buf);
//第一步,创建子进程
pid = fork();
if(pid < 0)
{
printf("error\n");
exit(1);//异常退出
}
else if(pid > 0)
{
exit(0);
}
//第二步,创建会话
setsid();
//第三步,修改根目录
chdir("/");
//第四步,设置掩码
umask(0);
for(i=0;i<MAXFILE;i++)
{
close(i);
}
while(1)
{
fd = open("/mnt/hgfs/H/C/dameon.log",O_CREAT|O_WRONLY,0600);
if(fd < 0)
{
perror("open");
exit(1);
}
write(fd,buf,len+1);
close(fd);
}
return 0;
}
我们执行ps -ef | grep dameon
识别二维码,关注我们
CPP后台服务器开发
linux|shell|C/C++|python