zoukankan      html  css  js  c++  java
  • 守护进程

    daemon进程步骤

    1.    调用fork产生一个子进程,同时父进程退出,所有后续工作都在子进程中完成。

    如果我们是从命令行执行的该程序,这可以造成程序执行完毕的假象,shell会回去等待下一条命令;
    刚刚通过fork产生的新进程一定不会是一个进程组的组长,这为第2步的执行提供了前提保障。

    由于父进程已经先于子进程退出,会造成子进程没有父进程,变成一个孤儿进程(orphan)。每当系统发现一个孤儿进程,就会自动由1号进程收养它,这样,原先的子进程就会变成1号进程的子进程。



    2.    调用setsid系统调用。

    这是整个过程中最重要的一步。它的作用是创建一个新的会话(session),并自任该会话的组长(session leader)。如果调用进程是一个进程组的组长,调用就会失败,但这已经在第1步得到了保证。

    调用setsid有3个作用:

    • 让进程摆脱原会话的控制;
    • 让进程摆脱原进程组的控制;
    • 让进程摆脱原控制终端的控制;

    setsid函数:一个会话(session)开始于用户登陆,终止于用户退出,在此期间该用户运行的所有进程都属于这个会话,除非进程调用setsid系统调用。系统调用setsid不带任何参数,调用之后,调用进程就会成立一个新的会话,并自任该会话的组长。

    总之,就是让调用进程完全独立出来,脱离所有其他进程的控制。



    3.    把当前工作目录切换到根目录。

    如果我们是在一个临时加载的文件系统上执行这个进程的,比如:/mnt/floppy/,该进程的当前工作目录就会是/mnt/floppy/。在整个进程运行期间该文件系统都无法被卸下(umount),而无论我们是否在使用这个文件系统,这会给我们带来很多不便。解决的方法是使用chdir系统调用把当前工作目录变为根目录,应该不会有人想把根目录卸下吧。
    当然,在这一步里,如果有特殊的需要,我们也可以把当前工作目录换成其他的路径,比如/tmp。



    4.    将文件权限掩码设为0。这需要调用系统调用umask。

    每个进程都会从父进程那里继承一个文件权限掩码,当创建新文件时,这个掩码被用于设定文件的默认访问权限,屏蔽掉某些权限,如一般用户的写权限。当另一个进程用exec调用我们编写的daemon程序时,由于我们不知道那个进程的文件权限掩码是什么,这样在我们创建新文件时,就会带来一些麻烦。所以,我们应该重新设置文件权限掩码,我们可以设成任何我们想要的值,但一般情况下,大家都把它设为0,这样,它就不会屏蔽用户的任何操作。
    如果你的应用程序根本就不涉及创建新文件或是文件访问权限的设定,你也完全可以把文件权限掩码一脚踢开,跳过这一步。



    5.    关闭所有不需要的文件。

    同文件权限掩码一样,我们的新进程会从父进程那里继承一些已经打开了的文件。这些被打开的文件可能永远不被我们的daemon进程读或写,但它们一样消耗系统资源,而且可能导致所在的文件系统无法卸下。需要指出的是,文件描述符为0、1和2的三个文件,也就是我们常说的输入、输出和报错这三个文件也需要被关闭。难道不需要输入输出吗?但事实是,在上面的第2步后,我们的daemon进程已经与所属的控制终端失去了联系,我们从终端输入的字符不可能达到daemon进程,daemon进程用常规的方法(如printf)输出的字符也不可能在我们的终端上显示出来。所以这三个文件已经失去了存在的价值,也应该被关闭。  

     

    http://topic.csdn.net/u/20110804/10/db36174c-4c05-405d-a77b-a079e3d7f623.html

    //把进程初始化为Daemon,返回0或1说明main需要结束,返回0x80表示可继续.int initDaemon(const char* lpcszMyName) { int i,pid,hFile; char szFilename[108]; //第一次fork pid = fork(); if (pid > 0) { //父进程到此结束. return 0; } if (pid < 0) { //fork失败,退出. return 1; } //////////////////////////////////// //只有第一子进程可以走到这里继续执行. //第一子进程成为新的会话组长和进程组长. setsid(); //第二次fork pid = fork(); if (pid > 0) { //第一子进程到此结束. return 0; } if (pid < 0) { //fork失败,退出. return 1; } //////////////////////////////////// //只有第二子进程可以走到这里继续执行. //与控制终端分离. for (i = 0; i < NOFILE; i++) { //关闭打开的文件描述符. close(i); } //改变工作目录到/tmp防止无法卸载文件系统. chdir("/tmp"); //重定向标准输出. sprintf(szFilename, "/tmp/%s.out", lpcszMyName); hFile = open(szFilename, O_WRONLY|O_CREAT|O_TRUNC, 0644); if (hFile != -1) { dup2(hFile, 1);//1号句柄代表标准输出设备. close(hFile); } //重定向标准错误. sprintf(szFilename, "/tmp/%s.err", lpcszMyName); hFile = open(szFilename, O_WRONLY|O_CREAT|O_TRUNC, 0644); if (hFile != -1) { dup2(hFile, 2);//2号句柄代表标准错误设备. close(hFile); } //重设文件创建掩码. umask(0); //处理SIGCHLD信号. signal(SIGCHLD, SIG_IGN); return 0x80; }

     

  • 相关阅读:
    Spring中 @PathVariable
    消息队列中点对点与发布订阅区别
    rabbitMQ下载地址
    当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递?
    String和StringBuilder、StringBuffer的区别?
    char 型变量中能不能存贮一个中文汉字,为什么?
    抽象类(abstract class)和接口(interface)有什么异同?
    静态嵌套类(Static Nested Class)和内部类(Inner Class)的不同?
    抽象的(abstract)方法是否可同时是静态的(static),是否可同时是本地方法(native),是否可同时被synchronized修饰?
    阐述静态变量和实例变量的区别。
  • 原文地址:https://www.cnblogs.com/cy568searchx/p/2730690.html
Copyright © 2011-2022 走看看