Linux中的守护进程

概念

Linux的守护进程又被称为Daemon进程,为Linux的后台进程(独立于终端)。一般来说,我们在程序中创建的进程会随着终端的关闭而退出。那么假如我们需要一个在Linux后台不断运行的进程该如何做呢?这个时候就需要我们的守护进程了。守护进程通常周期性地执行着某种任务或者等待着处理某些事件。其生命周期较长,通常在系统启动的时候开始执行,在系统关闭时终止,在Linux中的许多系统服务都是通过守护进程来实现的。一般的网络服务也以守护进程的形式来实现。那么,如何创建一个守护进程呢?

创建守护进程的步骤

  • 创建一个子进程,并让其成为孤儿进程
    在父进程中创建一个子进程,并让父进程先于子进程退出,此时,子进程便成为了一个孤儿进程。
  • 在子进程中创建新会话
    由于子进程在创建的时候复制了父进程的会话,进程组和终端控制等,虽然父进程退出了,但是原先的会话,进程组和终端控制等并没有改变,子进程还未完成真正的独立。在Linux系统中可以采用setsid()函数来创建一个新会话,并让进程担任该会话的组长,同时在会话组中创建新的进程组,该进程也是新进程组的组长。从而使得该进程成为新会话组跟新进程组中的唯一的进程,至此该进程完全脱离了终端的控制,运行在后台。
    setsid()函数原型如下
#include <unistd.h>
pid_t setsid();
  • 改变当前的工作目录
    由于采用fork()函数创建的子进程继承了父进程的当前工作目录,而系统通常的做法是让根目录成为守护进程的当前工作目录,所以我们也需要相应的改变子进程的当前工作目录,改变工作目录可以使用chdir()函数。
    函数原型如下:
#include <unistd.h>
int chdir(const char* path);
  • 重设文件权限掩码
    文件权限掩码的作用是屏蔽文件权限中的对应位。由于创建的子进程继承了父进程的文件权限掩码,这给子进程操作文件带来了一定的影响。因此通常需要把文件权限掩码设置为0,这样能够增强守护进程的灵活性。可以使用umask()改变文件权限掩码,参数为要修改的掩码值
    该函数原型如下:
#include <sys/types.h>
#include <sys/stat.h>

mode_t umask(mode_t mask);
  • 关闭文件描述符
    新创建的子进程会从父进程继承一些已经打开的文件描述符,而这些文件描述符可能永远都不会被子进程(守护进程)所访问,但它们却占用了一定的资源。所以,应该关闭,可以使用如下代码进行关闭:
int num = getdtablesize(); //获取当前进程文件描述符表的大小
for(int i = 0; i < num; ++i)
{
	close(i);
}

Demo

// this is daemon process example
//
#include <iostream>
#include <cstring>
using namespace std;

#include <sys/types.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/stat.h>
#include <stdlib.h>
#include <fcntl.h>

int main(int argc, const char** argv)
{
	pid_t pid;
	int i, fd;

	char* buf = "this is a Daemon\n";

	pid = fork();

	if(pid < 0)
	{
		cout << "fork error" << endl;
		return -1;
	}
	else if(pid > 0)
	{
		// father process
		exit(0);
	}
	else
	{
		// child process
		setsid();
		chdir("/tmp");
		umask(0);
		for(int i = 0; i < getdtablesize(); ++i)
		{
			close(i); //close the file description flag
		}

		if(fd = open("daemon.log", O_CREAT|O_WRONLY|O_TRUNC, 0600) < 0)
		{
			cout << "open error" << endl;
			return -1;
		}

		while(1)
		{
			write(fd, buf, strlen(buf));
			sleep(3);
		}
		close(fd);
	}
	return 0;
}

该段程序创建了一个守护进程,并执行每三秒向daemon.log文件输入字符串的任务。
在终端输入ps axj | grep ./daemon
可以看到
在这里插入图片描述
可以看到其进程ID,组ID和会话组ID保持一致,说明当前守护进程为组长,“?”代表其为后台进程。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值