创建守护进程

在linux或者unix操作系统中在系统的引导的时候会开启很多服务,这些服务就叫做守护进程。守护进程是在后台运行不与任何控制终端关联,是Linux中的后台服务进程。它是一个生存期较长的进程,通常独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。大多数服务都是通过守护进程实现的。它从被执行开始运转,直到整个系统关闭时才退出。如果想让某个进程不因为用户或终端或其他地变化而受到影响,那么就必须把这个进程变成一个守护进程。

通过一系列的操作,可以把一个普通进程转变位守护进程。

1.调用umask将文件模式创建屏蔽字设为0。

为了防止守护进程从继承来的文件模式创建屏蔽字屏蔽了某些权限。

2.调用fork,终止父进程。

如果进程是从一个shell命令启动的,父进程终止shell会认为命令执行完毕。

而子进程继承了父进程的进程组ID,而它的进程ID和父进程ID不同,这就保证  .子进程不是一个进程组的头进程,为调用setsid提供必要条件。

3.进程调用setsid建立一个会话。

首先要知道会话的概念,会话是一个多个进程组的集合。

如果该调用进程已经是一个进程组的组长进程该函数返回出错。所以通常先fork后使父进程终止。子进程再调用setsid.正如第一步。如下面的程序,父进程作为组长进程调用setsid将会报错。

main()
{
	int pid;
	if((pid = fork()) != -1)
	{
		if (pid != 0)
		{
			if ((pid = setsid()) == -1)
				perror("setsid error");
		}
	}

}

调用setsid后,调用进程成为新会话的第一个进程,此时该进程是新会话中唯一的进程,并且成为新进程组的组长进程,新进程组ID为调用进程ID。

4.忽略SIGHUP信号,再次调用fork,终止父进程。

确保子进程不再是一个会话头进程,不能自动获得一个控制终端。这里忽略SIGHUP信号是因为会话首进程终止时,会话中所有进程都会收到SIGHUP信号而终止。

5.改变工作目录。

当前工作目录可能是挂在到系统上的,如果守护进程运行,会导致挂载的目录无法卸载,所以要改变目录。

6.关闭不再需要的文件描述符。

调用sysconf(_SC_OPEN_MAX)获取进程可打开文件的限制,并关闭所有描述符。

7.将stdin,stdout,stderr重定向到/dev/null

如果stdin,stdout,stderr被关闭,如果建立了socket描述符,描述符值可能为stdin,stdout或stderr中值的某一个,那么当调用向标准输入输出出错写入数据的操作可能会发送给socket套接字对端,造成错误。

例如下面的程序,关闭标准出错,建立一个套接字,正常情况下该套接字描述符为2,即标准出错的描述符值,那么调用perror向标准出错中写入数据,将发送到对端,对端将接收到调用perror发送来的数据并打印,然后perror写入到标准出错的数据只被对端打印了一次,具体内部细节这里不做讨论。

关闭标准出错,调用perror的客户程序

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <stdlib.h>
#include <strings.h>
#include <string.h>

main()
{
	int r;
	int sockfd;
	struct sockaddr_in addr;

	//关闭标准出错
	close(2);
	sockfd = socket(AF_INET, SOCK_STREAM, 0);
	if (sockfd != 2)
	{
		printf("sockfd != 2\n");
		exit(-1);
	}
	bzero(&addr, sizeof(addr));
	addr.sin_family = AF_INET;
	addr.sin_port = htons(12345);
	addr.sin_addr.s_addr = inet_addr("127.0.0.1");

	r = connect(sockfd, (struct sockaddr*)&addr, sizeof(addr));
	if (r == -1)
	{
		printf("connect error\n");
		exit(-1);
	}

	//向标准出错中发送数据
	while (1)
	{
		sleep(2);
		perror("test");
	}
}

服务程序

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <stdlib.h>
#include <strings.h>

main()
{
	int r;
	int sockfd, clifd;
	struct sockaddr_in addr;
	char buf[1025];

	sockfd = socket(AF_INET, SOCK_STREAM, 0);
	bzero(&addr, sizeof(addr));
	addr.sin_family = AF_INET;
	addr.sin_port = htons(12345);
	addr.sin_addr.s_addr = inet_addr("127.0.0.1");

	r = bind(sockfd, (struct sockaddr*)&addr, sizeof(addr));
	if (r == -1)
	{
		printf("bind error\n");
		exit(-1);
	}

	r = listen(sockfd, 10);
	if (r == -1)
	{
		printf("listen error\n");
		exit(-1);
	}

	while(1)
	{
		clifd = accept(sockfd, NULL, NULL);
		if (clifd == -1)
		{
			printf("accept error\n");
			exit(-1);
		}

		r = read(clifd, buf, sizeof(buf));
		if (r == -1)
		{
			perror("error\n");
			exit(-1);
		}
		if (r == 0)
		{
			printf("对端关闭\n");
			exit(-1);
		}
		buf[r] = 0;
		printf("%s\n", buf);
	}
}

下面展示了一个创建守护进程的示例

#include <unistd.h>
#include <stdio.h>
#include <signal.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <syslog.h>

#define TEST_PID 1234

main()
{
	int pid;
	int i;
	long maxfd;

	//文件模式创建屏蔽字置0
	umask(0);

	pid = fork();
	if (pid == -1)
	{
		perror("fork1 error");
		exit(-1);
	}
	if (pid != 0)
		exit(0);

	//子进程1继续
	if (setsid() == -1)
	{
		perror("setsid error");
		exit(-1);
	}
	signal(SIGHUP, SIG_IGN);
	pid = fork();
	if (pid == -1)
	{
		perror("fork2 error");
		exit(-1);
	}
	if (pid != 0)
		exit(0);

	//子进程2继续
	//改变工作目录
	chdir("/");

	//关闭描述符
	maxfd = sysconf(_SC_OPEN_MAX);
	for (i=0; i<maxfd; i++)
		close(i);

	//stdin,stdout,stderr重定向到/dev/null
	open("/dev/null", O_RDONLY);
	open("/dev/null", O_RDWR);
	open("/dev/null", O_RDWR);
}

创建守护进程,布布扣,bubuko.com

时间: 2024-10-14 00:17:58

创建守护进程的相关文章

ASP.ENT Core Linux 下 为 donet创建守护进程(转载)

原文地址:http://www.cnblogs.com/savorboard/p/dotnetcore-supervisor.html 前言 在上篇文章中介绍了如何在 Docker 容器中部署我们的 asp.net core 应用程序,本篇主要是怎么样为我们在 Linux 或者 macOs 中部署的 dotnet 程序创建一个守护进程,来保证我们的程序在异常或者是电脑重启的时候仍然能够正常访问. 如果你以后用准备使用 asp.net core来开发项目的话,程序并且部署到 Linux 上的话,那

linux创建守护进程

守护进程deamon 是一个后台进程,无需用户输入就能运行,用来在系统后台提供某种服务. 常见的守护进程有Wbe服务器.邮件服务器以及数据库服务器等等.守护进程不能控制终端,所以任何的输入和输出都需要做特殊处理. 创建守护进程的过程并不复杂,首先执行fork(),将父进程退出. 一个守护进程是从shell脚本或者命令启动,守护进程和应用程序不同的地方是守护进程不是交互式的,它在后台运行,不控制终端. 守护进程既不需要从标准输入设备读入信息,也不需要从标准输出设备输出信息. 下一步是在子进程中调用

创建守护进程步骤与setsid() -- linux deamon进程

原创:http://www.cnblogs.com/mickole/p/3188321.html 一,守护进程概述 Linux Daemon(守护进程)是运行在后台的一种特殊进程.它独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件.它不需要用户输入就能运行而且提供某种服务,不是对整个系统就是对某个用户程序提供服务.Linux系统的大多数服务器就是通过守护进程实现的.常见的守护进程包括系统日志进程syslogd. web服务器httpd.邮件服务器sendmail和数据库服务器mys

linux 创建守护进程的相关知识

linux 创建守护进程的相关知识 http://www.114390.com/article/46410.htm linux 创建守护进程的相关知识,这篇文章主要介绍了linux 创建守护进程的相关知识,需要的朋友可以参考下 关键字:linux.守护进程 创建子进程,父进程退出 这是编写守护进程的第一步.由于守护进程是脱离控制终端的,因此,完成第一步后就会在Shell终端里造成一程序已经运行完毕的假象.之后的所有工作都在子进程中完成,而用户在Shell终端里则可以执行其他命令,从而在形式上做到

创建守护进程-转

链接:https://www.zhihu.com/question/38609004/answer/77190522 守护进程是在后台运行不受终端控制的进程(如输入.输出等),一般的网络服务都是以守护进程的方式运行.守护进程脱离终端的主要原因有两点:(1)用来启动守护进程的终端在启动守护进程之后,需要执行其他任务.(2)(如其他用户登录该终端后,以前的守护进程的错误信息不应出现)由终端上的一些键所产生的信号(如中断信号),不应对以前从该终端上启动的任何守护进程造成影响.要注意守护进程与后台运行程

C语言创建守护进程

守护进程最大的特点就是不受终端控制,并且在后台运行.linux上很多应用都是以守护进程的方式存在的,比如常用的crond.把应用做成守护进程的好处就是可以脱离终端,不会因为终端的关闭而挂掉,会随着系统的存在而存在. 创建守护进程有以下几个要点: <1> 创建子进程,杀掉父进程. 这么做主要是为了确保进行操作<2>的进程不是会话组长,因为会话组组长调用setsid会失败. <2> 调用setsid() 每个进程都属于一个进程组,进程组号就是进程组组长的pid.多个进程组属

Supervisor 为服务创建守护进程

今天需要再服务上部署一个.net 方面的项目:当时开启服务的命令只能在前台执行:使用nohub CMD &等放在后台开启服务都会宕机:所以搜寻了Supervisor 这个解决办法,为服务创建守护进程.具体操作如下 1.什么是守护进程 在linux或者unix操作系统中,守护进程(Daemon)是一种运行在后台的特殊进程,它独立于控制终端并且周期性的执行某种任务或等待处理某些发生的事件.由于在linux中,每个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程都会依附于这个终端,这个终

ASP.NET Core Linux下为 dotnet 创建守护进程(必备知识)

原文:ASP.NET Core Linux下为 dotnet 创建守护进程(必备知识) 前言 在上篇文章中介绍了如何在 Docker 容器中部署我们的 asp.net core 应用程序,本篇主要是怎么样为我们在 Linux 或者 macOs 中部署的 dotnet 程序创建一个守护进程,来保证我们的程序在异常或者是电脑重启的时候仍然能够正常访问. 如果你以后用准备使用 asp.net core来开发项目的话,程序并且部署到 Linux 上的话,那么此篇文章你值得收藏. 如果你觉得对你有帮助的话

centos7用shell创建守护进程,并设置为系统服务,开机自动启动

首先是创建shell守护进程:(shouhu.sh) #!/bin/bash test(){         while true         do                 ls ./  >> 123.txt                 sleep 5                              done } case $1 in start)         path="/root/shouhu.pid"         if [ -f $p