linux将一个进程变成守护进程,Linux进程实践(5) --守护进程

概述

守护进程是在需要在后台长期运行不受终端控制的进程,通常情况下守护进程在系统启动时自动运行,在服务器关闭的时候自动关闭;守护进程的名称通常以d结尾,比如sshd、xinetd、crond、atd等。

守护进程编程规则

调用umask将文件模式创建屏蔽字设置为一个已知值(通常是0)

调用fork(),创建新进程,它会是将来的守护进程

然后使父进程exit,保证子进程不是进程组组长

调用setsid创建新的会话

会话:是一个或者多个进程组的集合,通常一个会话开始与用户登录,终止于用户退出。在此期间,该用户运行的所有进程都属于这个会话期。

将进程的当前目录改为根目录 (如果把当前目录作为守护进程的目录,当前目录不能被卸载,它作为守护进程的工作目录了。)

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

将标准输入、标准输出、标准错误重定向到/dev/null

setsid

pid_t setsid(void);

setsid() creates a new session if the calling process is not a process group leader.  Thecalling processis the leader of thenew session, the process group  leader  of  the  new process  group,  and has no controlling terminal.  The process group ID and session ID of the calling process are set to the PID of the calling process.  The calling process  will be the only process in this new process group and in this new session.

/*当调用进程不是一个进程组的组长时,Setsid创建一个新的会话;调用者进程会是这个会话期唯一的一个进程,且是该进程组的组长;调用者进程id是组id,也是会话期的id。不能用进程组组长去调用setsid函数*/

//示例:

int main()

{

pid_t pid = fork();

if (pid == -1)

err_exit("fork error");

else if (pid != 0)

exit(EXIT_SUCCESS);

// //查看下面这一部分代码在注释的前后有什么变化

// pid_t id = setsid();

// if (id == -1)

// err_exit("setsid error");

// else

// cout << "new session id = " << id << endl;

cout << "getpid = " << getpid() << endl;

cout << "getpgid = " << getpgid(getpid()) << endl;

return 0;

}

RETURN VALUE

On  success,  the  (new)  session  ID  of  the  calling  process  is returned.  On error, (pid_t) -1 is returned, and errno is set to indicate the error.

Linux中的守护进程API

int daemon(int nochdir, int noclose);

参数:

nochdir:=0将当前目录更改至“/”

noclose:=0将标准输入、标准输出、标准错误重定向至“/dev/null”

DESCRIPTION

The  daemon()  function is for programs wishing to detach themselves from the controlling terminal and run in the background as system daemons. If nochdir is zero, daemon() changes the calling process's current working  directory  to the root directory ("/"); otherwise, the current working directory is left unchanged. If noclose is zero, daemon() redirects standard input, standard output and standard error to /dev/null; otherwise, no changes are made to these file descriptors.

//示例:自己动手写daemon函数(一个比较简单的示例)

bool myDaemon(bool nochdir, bool noclose)

{

umask(0);

pid_t pid = fork();

if (pid == -1)

err_exit("fork error");

else if (pid != 0) //parent

exit(0);

setsid();

if (nochdir == 0)

chdir("/");

if (noclose == 0)

{

int i;

for (i=0; i < 3; ++i)

close(i);

open("/dev/null", O_RDWR); //相当于把0号文件描述符指向/dev/null

dup(0); //把0号文件描述符 赋值给空闲的文件描述符 1

dup(0); //把0号文件描述符 赋值给空闲的文件描述符 2

}

return true;

}

//测试

int main(int argc, char *argv[])

{

myDaemon(0, 0); //0表示做出改变(当前目录,文件描述符),1表示不改变

printf("test ...\n");

while (true)

{

sleep(1);

}

return 0;

}

kclIlEnaWWmR6OaL.png

//一个比较经典和完善的实例;来自《APUE》

#include "apue.h"

#include

#include

#include

bool myDaemon(const char *cmd);

int main(int argc, char *argv[])

{

myDaemon("xiaofang");

while (true)

{

sleep(1);

}

return 0;

}

bool myDaemon(const char *cmd)

{

umask(0);

//Get maximum number of file descriptors.

rlimit rlt;

if (getrlimit(RLIMIT_NOFILE,&rlt) < 0)

{

err_quit("%s: can't get file limit",cmd);

}

//Become a session leader to lose controlling TTY.

pid_t pid = fork();

if (pid == -1)

{

err_quit("%s: can't fork",cmd);

}

if (pid != 0) //parent

{

exit(0);

}

setsid();

//Ensure future opens won't allocate controlling TTYs.

struct sigaction sa;

sa.sa_handler = SIG_IGN;

sigemptyset(&sa.sa_mask);

sa.sa_flags = 0;

if (sigaction(SIGHUP,&sa,NULL) < 0)

{

err_quit("%s can't ignore SIGHUP",cmd);

}

if ((pid = fork()) < 0)

{

err_quit("%s: can't fork",cmd);

}

else if (pid != 0) //Second Parent

{

exit(EXIT_SUCCESS);

}

//change the current working directory to the root

if (chdir("/") < 0)

{

err_quit("%s: can't change directory to /",cmd);

}

//close all open file description

if (rlt.rlim_max == RLIM_INFINITY)

{

rlt.rlim_max = 1024;

}

for (unsigned int i = 0; i < rlt.rlim_max; ++i)

{

close(i);

}

//Attach file descriptors 0, 1, and 2 to /dev/null.

int fd0 = open("/dev/null",O_RDWR);

int fd1 = dup(0);

int fd2 = dup(0);

//Initialize the log file.

openlog(cmd,LOG_CONS,LOG_DAEMON);

if (fd0 != 0 || fd1 != 0 || fd2 != 0)

{

syslog(LOG_ERR,"unexpected file descriptors %d %d %d",

fd0,fd1,fd2);

exit(EXIT_FAILURE);

}

return true;

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值