迹忆客 专注技术分享

当前位置:主页 > 学无止境 > 编程语言 > C语言 >

在 C 语言中从另一个进程控制守护进程

作者:迹忆客 最近更新:2023/03/28 浏览次数:

本文将介绍有关如何从 C 语言中的另一个进程控制守护进程的多种方法。


使用 forksetsid 函数创建守护进程

守护程序进程具有多个特征,例如它们是长时间运行的进程,并且守护程序可以在系统启动时启动。守护进程可以从用户命令中进行控制,这些命令可以强制守护进程终止或暂停,甚至在启动时禁用。虽然,常见的情况是需要使用某些特定于系统的脚本的守护程序在系统关闭时终止。守护程序通常在没有控制终端的情况下在后台运行,采用这种功能的程序应实现各种信号处理程序以处理外部中断。

有多种方法可以创建守护进程并对其进行监视,但是在此示例中,我们演示了创建阶段,在此阶段中,我们调用 fork 函数来创建子进程。然后,父进程退出,子进程继续成为 init 进程的子进程(在 Linux 系统上,init 进程是启动时的第一个进程),子进程继续执行。子进程调用 setsid 函数以启动新会话并从控制终端中删除该进程。最后,我们再次调用 fork 并退出父进程,以确保我们的守护程序不会获得控制终端。现在我们正在执行守护进程;重要的是注册一个 SIGTERM 信号处理程序,以便在系统或用户传递相应的中断时进行任何资源清除和正常退出。

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <signal.h>

#define errExit(msg) do { perror(msg); exit(EXIT_FAILURE); \
                               } while (0)

void cleanupRoutine(int signal_number)
{
    write(STDERR_FILENO, "hello", 5);
    _exit(EXIT_SUCCESS);
}

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

    fprintf(stderr,"[pid - %d] running...\n", getpid());

    switch (fork()) {
        case -1: errExit("fork");
        case 0:  break;
        default: _exit(EXIT_SUCCESS);
    }
    fprintf(stderr,"[pid - %d] running...\n", getpid());

    if (setsid() == -1)
        errExit("setsid");

    switch (fork()) {
        case -1: errExit("fork");
        case 0:  break;
        default: _exit(EXIT_SUCCESS);
    }
    fprintf(stderr,"[pid - %d] running...\n", getpid());

    struct sigaction sigterm_action;

    memset(&sigterm_action, 0, sizeof(sigterm_action));
    sigterm_action.sa_handler = &cleanupRoutine;
    sigterm_action.sa_flags = 0;

    // Mask other signals from interrupting SIGTERM handler
    if (sigfillset(&sigterm_action.sa_mask) != 0) {
        perror("sigfillset");
        exit(EXIT_FAILURE);
    }
    // Register SIGTERM handler
    if (sigaction(SIGTERM, &sigterm_action, NULL) != 0) {
        perror("sigaction SIGTERM");
        exit(EXIT_FAILURE);
    }

    while (1) {
        getpid();
    }

    exit(EXIT_SUCCESS);
}

使用 daemon 函数创建守护进程

即使前面的示例演示了看似正确的代码,也缺少确保关闭从父级继承的所有打开文件描述符的步骤;工作目录已更改,将标准输入重定向到/dev/null,依此类推。这些步骤可确保某些功能在守护程序调用时不会失败,并且不会观察到某些奇怪的行为。

例如,如果你从终端窗口启动了先前的程序,然后将 SIGTERM 信号发送到守护进程,那么 cleanupRoutine 信号处理程序中的 write 函数即使在显示新的提示后也会打印到同一个终端。因此,GNU C 库提供了 daemon 函数。它会自动执行上述步骤,以确保新创建的守护进程具有干净的上下文。daemon 函数采用两个整数参数:第一个(如果等于零)指定是否应将当前工作目录更改为根目录。第二个整数(如果等于零)指示是否将标准 I/O 流重定向到/dev/null

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <signal.h>

#define errExit(msg) do { perror(msg); exit(EXIT_FAILURE); \
                               } while (0)

void cleanupRoutine(int signal_number)
{
    write(STDERR_FILENO, "hello", 5);
    _exit(EXIT_SUCCESS);
}

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

    fprintf(stderr,"[pid - %d] running...\n", getpid());

    if (daemon(0, 0) == -1)
        errExit("daemon");

    fprintf(stderr,"[pid - %d] running...\n", getpid());

    struct sigaction sigterm_action;

    memset(&sigterm_action, 0, sizeof(sigterm_action));
    sigterm_action.sa_handler = &cleanupRoutine;
    sigterm_action.sa_flags = 0;

    // Mask other signals from interrupting SIGTERM handler
    if (sigfillset(&sigterm_action.sa_mask) != 0)
        errExit("sigfillset");

    // Register SIGTERM handler
    if (sigaction(SIGTERM, &sigterm_action, NULL) != 0)
        errExit("sigaction");

    while (1) {
        getpid();
    }

    exit(EXIT_SUCCESS);
}

转载请发邮件至 1244347461@qq.com 进行申请,经作者同意之后,转载请以链接形式注明出处

本文地址:

相关文章

在 C 语言中使用 typedef enum

发布时间:2023/05/07 浏览次数:181 分类:C语言

本文介绍了如何在 C 语言中使用 typedef enum。使用 enum 在 C 语言中定义命名整数常量 enum 关键字定义了一种叫做枚举的特殊类型。

C 语言中的静态变量

发布时间:2023/05/07 浏览次数:151 分类:C语言

本文介绍了如何在 C 语言中使用静态变量。在 C 语言中使用 static 变量在函数调用之间保存变量值

C 语言中生成随机数

发布时间:2023/05/07 浏览次数:64 分类:C语言

本文演示了如何在 C 语言中生成随机数。使用 rand 和 srand 函数在 C 语言中生成随机数

C 语言中的 i++ 与++i

发布时间:2023/05/07 浏览次数:83 分类:C语言

本文演示了如何在 C 语言中使用前缀增量与后缀增量运算符。C 语言中++i 和 i++ 记号的主要区别

C 语言中获取当前工作目录

发布时间:2023/05/07 浏览次数:182 分类:C语言

本文演示了如何在 C 语言中获取当前工作目录。使用 getcwd 函数获取当前工作目录的方法

C 语言中的位掩码

发布时间:2023/05/07 浏览次数:126 分类:C语言

本文介绍了如何在 C 语言中使用位掩码。使用 struct 关键字在 C 语言中定义位掩码数据

C 语言中的排序函数

发布时间:2023/05/07 浏览次数:181 分类:C语言

本文演示了如何在 C 语言中使用标准库排序函数。使用 qsort 函数对 C 语言中的整数数组进行排序

C 语言中的 extern 关键字

发布时间:2023/05/07 浏览次数:114 分类:C语言

本文介绍了如何在 C 语言中使用 extern 关键字。C 语言中使用 extern 关键字来声明一个在其他文件中定义的变量

C 语言中的 #ifndef

发布时间:2023/05/07 浏览次数:186 分类:C语言

本文介绍了如何在 C 语言中使用 ifndef。在 C 语言中使用 ifndef 保护头文件不被多次包含 C 语言中的头文件用于定义同名源文件中实现的函数的接口。

扫一扫阅读全部技术教程

社交账号
  • https://www.github.com/onmpw
  • qq:1244347461

最新推荐

教程更新

热门标签

扫码一下
查看教程更方便