为了创建一个守护进程(至少在 Linux 上),必须采取一些步骤,我现在将对其进行描述。在问题的最后,如果有人明白,那么请回答,我将不胜感激。
1)您需要调用一个函数fork()来创建子进程并终止父进程。这样做是为了让我们的子进程在后台混淆(如果错误,则正确)
if( (pid = fork()) < 0)
return -1;
else if(pid)
exit(0);
2)然后我们调用setsid()创建新会话的函数。另外,调用进程成为组的leader,新会话的leader,没有控制终端。
if(setsid() < 0)
return -1;
"Разработка сетевых приложений" Стивенс У.Р在我举这个例子的书中,作者再次调用该函数fork()以确保守护程序在打开终端设备时不能自动获取控制终端。请注意,当父进程终止时(在我们的例子中первого дочернего процесса),会向会话中的所有进程(包括我们的второму дочернему процессу)发送一个信号SIGHUP(当有人杀死终端,而不杀死终端窗口内运行的应用程序时,操作系统将信号发送给程序) ,我们需要忽略它。
3)我们用命令改变当前目录chdir("/");
4) 关闭文件描述符。作者是这样做的
for(i = 0; i < 64; i++)
close(i);
还有更多...我不会给出其余的代码,因为它与问题的本质无关,但是我之前指出的互联网上有一本书,您可以在其中找到完整的示例。
事实上,我的вопросы:
1)这个问题可能看起来很傻,但我不太明白什么управляющий терминал。是不是像下面的终端root?请解释它是什么以及它的作用。
2)为什么需要更改当前目录?书上说的不多,我也不是很懂。
3)我们关闭负责的描述符是什么?他们应该完全关闭吗?
简而言之,这是与当前控制台会话 ( session ) 关联的终端。列在
TTYps“a”列中。事实上,它相当于/dev/tty*“真实”控制台的设备之一,或 /dev/pts* 用于伪终端(unix98)。还有一个特殊的设备/dev/tty是指当前进程的控制端。这种抽象的主要目的是在进程组之间切换——其中只有一个可以在前台,其余的在后台。此外,UT(在规范模式下)负责在输入出现特殊字符(例如
^C,或^Z)时向前台进程发送信号。另外,在UT的帮助下,程序可以直接向用户请求一些数据,比如密码,当I/O被重定向时,尤其ssh是sudo.一般来说,在 UNIX 中,作业管理,实际上是与终端相关的整个子系统,是一个相当混乱的东西,有许多大大小小的颠簸,其中大部分是受历史启发和使一切变得灵活的需要,但同时干净,因此对目的和 UT 细节的描述,以及诸如“进程组”、“会话”、“会话领导”等抽象,值得一本书或至少一篇文章单独列出。
原则上,这不是必需的,但最好释放未使用的资源。例如,如果不这样做,那么将无法使用用户启动守护程序的目录卸载文件系统。
与上述相同,不是必需的,但通常是很好的做法。前三个是标准输入/输出/错误流,其余的是其他可以打开的句柄——各种套接字、文件等。
不幸的是,POSIX 中没有简单且可移植的方法允许您简单地“关闭所有打开的句柄”甚至“查询所有打开的描述符”,因此通常的做法是遍历它们,从零到最后一个,并且关。是的,不依赖“64”魔术常数的更便携的选项是:
在 BSD 世界中,有一个相当方便的调用
closefrom ()可以关闭指定句柄之上的所有句柄。一些系统有类似的调用fcntl-F_CLOSEM.在 Linux 和其他一些系统上,您可以改为查看 /proc/self/fd (或
/dev/fd/*)的内容并仅关闭那些真正打开的内容。此处列出了有关此动物园的其他说明。例如,可以看到一个使用大量条件编译支持各种系统的真实示例
openssh。另外,前三个描述符(stdin、stdout、stderr)应该重新打开,例如on/dev/null,以免以后出错。
其他备注
“恶魔化”算法,省略错误处理等,可以写成 4 行。而且,只有第一个,部分第二个与妖魔化直接相关,其余的只是一个可取的擦除。