最近在使用daemon函数的时候,遇到了一个不算问题的问题,一开始没有发现也是由于我对这个函数的参数没有深入了解过。下面来稍微深入了解一下这个函数:
linux提供了daemon函数⽤于创建守护进程,使进程脱离控制台,在后台运行。
函数原型:
头文件
#include <unistd.h>
int daemon(int nochdir, int noclose);
- 当nochdir为0时,daemon将更改进程的根⽬录为root(“/”)。(我就是在这里犯的错误)
- 当noclose为0时,daemon将进城的STDIN, STDOUT, STDERR都重定向到/dev/null,也就是不输出任何信息,否则照样输出。一般情况下,这个参数都是设为0的。
如果程序在后台运行时有误,我们可以暂时把noclose设置为 1 来调试程序(一般情况下在控制台运行没问题的话,后台也不会有问题)。
deamon()调用了fork(),如果fork成功,父进程在daemon函数运行完毕后自杀,子进程由init进程领养。
我犯的错:
我在使用这个函数时,没有多想,两个参数都设置成的 0 ,导致程序原本在控制台运行得好好的,但是放到后台就会出错。
我在main函数里调用的一个函数,里面会创建一个文件,在对创建文件的返回值判断失败进行操作的时候,我直接exit(0)了。导致程序直接退出了。那么文件为什么会创建失败呢?
这就跟daemon函数的第一个参数有关了,因为当你设置参数为0时,daemon已经将进程的根目录改为root(“/”)了,而在root下,是无法创建文件的。所以我们在使用daemon函数的时候,没什么特殊需求的话,参数的设置最好是daemon(1, 0)。
daemon的实现⼤致如下:
int daemon( int nochdir, int noclose )
{
pid_t pid;
if ( !nochdir && chdir("/") != 0 ) //如果nochdir=0,那么改变到"/"根⽬录
return -1;
if ( !noclose ) //如果没有noclose标志
{
int fd = open("/dev/null", O_RDWR);
if ( fd < 0 )
return -1;
/* 重定向标准输⼊、输出、错误到/dev/null,键盘的输⼊将对进程⽆任何影响,
进程的输出也不会输出到终端*/
dup(fd, 0);
dup(fd, 1);
dup(fd, 2);
close(fd);
}
pid = fork(); //创建⼦进程.
if (pid < 0) //失败
return -1;
if (pid > 0)
_exit(0); //返回执⾏的是⽗进程,那么⽗进程退出,让⼦进程变成真正的孤⼉进程.
//创建的 daemon⼦进程执⾏到这⾥了
if ( setsid() < 0 ) //创建新的会话,并使得⼦进程成为新会话的领头进程
return -1;
return 0; //成功创建daemon⼦进程
}
链接:https://wenku.baidu.com/view/21534fb668ec0975f46527d3240c844769eaa064.html
如何杀死这样的进程:
通过ps aux | grep xxx找到对应的后台进程的进程ID,使⽤kill命令将进程杀死。