• 守护进程解析


    什么是守护进程? - 知乎

    什么是守护进程:生存期长的一种进程,没有控制终端。它们常常在系统引导装入时启动,仅在系统关闭时才终止。

    进程组 :

    • 每个进程除了有一个进程ID之外,还属于一个进程组
    • 进程组是一个或多个进程的集合,同一进程组中的各进程接收来自同一终端的各种信号
    • 每个进程组有一个组长进程。组长进程的进程组ID等于其进程ID

    会话:会话(session)是一个或多个进程组的集合,进程调用 setsid 函数(原型:pid_t setsid(void) )建立一个会话。

    进程调用 setsid 函数建立一个新会话,如果调用此函数的进程不是一个进程组的组长,则此函数创建一个新会话。具体会发生以下3件事:

    • 该进程变成新会话的会话首进程(session leader,会话首进程是创建该会话的进程)。此时,该进程是新会话的唯一进程。
    • 该进程成为一个新进程组的组长进程。新进程组ID是该调用进程的进程ID
    • 该进程没有控制终端。如果调用setsid之前该进程有一个控制终端,那么这种联系也被切断



    如果该调用进程已经是一个进程组的组长,则此函数返回出错。为了保证不处于这种情况,通常先调用fork,然后使其父进程终止,而子进程则继续。因为子进程继承了父进程的进程组ID,而其进程ID重新分配的,两者不可能相等,这就保证了子进程不是一个进程组的组长

    1. 作者:Zyoung
    2. 链接:https://www.zhihu.com/question/38609004/answer/529315259
    3. 来源:知乎
    4. 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
    5. void
    6. daemonize(const char *cmd)
    7. {
    8. int i, fd0, fd1, fd2;
    9. pid_t pid;
    10. struct rlimit rl;
    11. struct sigaction sa;
    12. /*
    13. * Clear file creation mask.
    14. */
    15. umask(0);
    16. /*
    17. * Get maximum number of file descriptors.
    18. */
    19. if (getrlimit(RLIMIT_NOFILE, &rl) < 0)
    20. err_quit("%s: can't get file limit", cmd);
    21. /*
    22. * Become a session leader to lose controlling TTY.
    23. */
    24. if ((pid = fork()) < 0)
    25. err_quit("%s: can't fork", cmd);
    26. else if (pid != 0) /* parent */
    27. exit(0);
    28. setsid();
    29. /*
    30. * Ensure future opens won't allocate controlling TTYs.
    31. */
    32. sa.sa_handler = SIG_IGN;
    33. sigemptyset(&sa.sa_mask);
    34. sa.sa_flags = 0;
    35. if (sigaction(SIGHUP, &sa, NULL) < 0)
    36. err_quit("%s: can't ignore SIGHUP", cmd);
    37. if ((pid = fork()) < 0)
    38. err_quit("%s: can't fork", cmd);
    39. else if (pid != 0) /* parent */
    40. exit(0);
    41. /*
    42. * Change the current working directory to the root so
    43. * we won't prevent file systems from being unmounted.
    44. */
    45. if (chdir("/") < 0)
    46. err_quit("%s: can't change directory to /", cmd);
    47. /*
    48. * Close all open file descriptors.
    49. */
    50. if (rl.rlim_max == RLIM_INFINITY)
    51. rl.rlim_max = 1024;
    52. for (i = 0; i < rl.rlim_max; i++)
    53. close(i);
    54. /*
    55. * Attach file descriptors 0, 1, and 2 to /dev/null.
    56. */
    57. fd0 = open("/dev/null", O_RDWR);
    58. fd1 = dup(0);
    59. fd2 = dup(0);
    60. /*
    61. * Initialize the log file.
    62. */
    63. openlog(cmd, LOG_CONS, LOG_DAEMON);
    64. if (fd0 != 0 || fd1 != 1 || fd2 != 2) {
    65. syslog(LOG_ERR, "unexpected file descriptors %d %d %d",
    66. fd0, fd1, fd2);
    67. exit(1);
    68. }
    69. }

     关于sighup是何时产生的:

    SIGHUP会在以下3种情况下被发送给相应的进程:
      1、终端关闭时,该信号被发送到session首进程以及作为job提交的进程(即用 & 符号提交的进程);
      2、session首进程退出时,该信号被发送到该session中的前台进程组中的每一个进程;
       3、若父进程退出导致进程组成为孤儿进程组,且该进程组中有进程处于停止状态(收到SIGSTOP或SIGTSTP信号),该信号会被发送到该进程组中的每一个进程

    详细可看:https://www.cnblogs.com/klb561/p/12051027.html

  • 相关阅读:
    MySQL的三种日志文件
    SpringMVC13-注解配置SpringMVC
    sql注入的数据提交方式和查询方式
    Feign远程调用丢失请求头
    10-Mysql内核查询成本计算实战-06
    Kotlin中的步长
    2022双十一买什么好?行家推荐四大最值得入手的数码好物
    【MATLAB教程案例45】基于双目视觉的图像深度信息提取算法matlab仿真
    C语言实验十一 指针(一)
    OpenCV实现图像的礼帽和黑帽
  • 原文地址:https://blog.csdn.net/m0_63691156/article/details/133497320