如何在Linux下自定义/编写 一个守护进程

【摘要】本文主要讲述何为守护进程,以及如何自编或者利用现有程序将其伪装成守护进程。

1、何为守护进程?

  • 正常情况下,当我们运行一个前台或后台进程时,一旦离开当前会话(终端),那该会话中的所有前后台进程也随即结束,当你重新打开会话时,已经“物是人非,难遇故人”了。
  • 而守护进程就可以不受会话的限制,可在前后台一直运行直至结束的进程。
  • 守护进程的实现有两种方式:自编和利用现有程序伪装,下文将分别讲述。

2、自编守护进程的一般步骤及代码实例

通常,当我们写一个守护进程时,需要按照如下步骤来写:

  1. 创建一个子进程,父进程随即退出

    • fork()函数用来创建一个子进程;

    • 在父进程退出后,后续所有步骤在子进程中进行。

  2. 在子进程中创建新会话

    • setsid()函数用来创建一个新的会话;
    • 创建新的会话的目的是使子进程完全从父进程中独立出来,使得即使父进程所在会话(终端)被关闭,也不会影响子进程。
  3. 改变子进程当前工作目录,一般设为根目录

    • chdir()函数用来为子进程重新分配工作目录;
    • 重新制定工作目录的原因是防止原来父进程工作目录含有可卸载的目录(文件系统),从而为今后守护进程的运行产生不必要的麻烦。
    • 工作目录一般设为根目录,当然也可以换成其它安全路径。
  4. 重设文件权限掩码

    • umask()函数用来更改子进程的默认掩码;
    • 文件权限掩码一般设为0,便于在守护进程中指定的文件权限不受掩码干扰,增加灵活性;
      • 通常,系统默认的文件权限为666(rw-rw-rw-),默认的文件权限掩码为022,故最后二者相与,得到的文件真实权限为644(rw-r–r–)。
  5. 关闭文件描述符

    • close()函数用来关闭文件描述符指定的文件;

    • 从父进程继承的打开文件及默认打开的0、1、2在守护进程中一般不会用到,而且又浪费系统资源,所以一般要关闭他们。

    • getdtablesize()函数返回当前打开的文件项数;

  6. 守护进程功能实现(无限循环)

  • 代码实例

    #include <stdio.h>  
    #include <stdlib.h>  
    #include <string.h>  
    #include <unistd.h>  
    #include <sys/wait.h>  
    #include <sys/types.h>  
    #include <fcntl.h>  
      
    int main() 
    {    
        pid_t pid;  
        int i, fd, len;  
        char *buf = "守护进程运行中.\n";  
        len = strlen(buf)+1;
        
        pid = fork();	//1.1 创建子进程
        if (pid < 0) {  
        	printf("fork error!");  
        	exit(1);  
        }
        if (pid>0) 		// 1.2父进程退出  
        	exit(0);  
          
        setsid(); 		// 2.在子进程中创建新会话。  
        
        chdir("/"); 	// 3.设置工作目录为根目录  
        
        umask(0); 		// 4.设置权限掩码  
        
        for(i=0; i<getdtablesize(); i++) //5.关闭用不到的文件描述符  
        	close(i);
        
        //6.守护进程功能实现
        while(1) {			// 死循环表征它将一直运行
            fd = open("/var/log/daemon.log",O_CREAT|O_WRONLY|O_APPEND,0600);
        	if(fd < 0) {
                printf("Open file failed!\n");
                exit(1);  
        	}  
        	write(fd, buf, len);  // 将buf写到fd中  
        	close(fd);  
        	sleep(10);  
        }  
        printf("error: Never run here!\n");  
    	return 0;  
    }
    
    • 注意:因为守护进程关闭了所有文件描述符,包括默认的标准输入输出和出错,所以在终端上看不到其输出。
    • 运行该程序后,可以通过执行:cat /var/log/daemon.log查看进程运行情况;也可以通过ps -axjf | grep 程序名查看。
    • 关闭当前会话(终端),再次重复执行上步命令,查看进程是否还在运行!

2.1 守护进程的出错处理

由于守护进程脱离了控制终端,因此,不能像其它普通进程一样将信息输出到控制终端来通知程序员。所以,只有使用syslog服务,将程序中的出错信息输入到系统日志文件中,例如/var/log/messages

syslog是Linux的系统日志管理服务,通过守护进程syslogd来维护。该服务在启动时会读配置文件/etc/syslog.conf,该文件决定了不同类别的消息会发往何处。例如,紧急消息发到控制台上直接显示,警告消息则发到记录文件中。

该机制为程序员提供了3个接口函数,分别为openlog()、syslog()、closelog():

  • openlog()

    • 作用:打开系统日志服务的一个连接
    • 原型:void openlog(char *ident, int option, int facility);
    • 参数:
      • ident:要向每个消息加入的字符串,通常为程序的名称
      • option:
        • LOG_CONS:若消息无法送到系统日志服务,则直接输出到系统控制终端
        • LOG_NDELAY:立即打开系统日志服务的连接。正常情况下,直到发送第一条消息时才打开连接
        • LOG_PERROR:将消息同时送到标准输出上
        • LOG_PID:在每条消息中包含进程的PID
      • facility:指定程序发送的消息类型:
        • LOG_AUTHPRIV:授权信息
        • LOG_CRON:时间守护进程(cron及at)
        • LOG_DAEMON:其它系统守护进程
        • LOG_KERN:内核信息
        • LOG_LOCAL[0~7]:保留
        • LOG_LPR:行打印机子系统
        • LOG_MAIL:邮件子系统
        • LOG_NEWS:新闻子系统
        • LOG_SYSLOG:syslog内部所产生的信息
        • LOG_USER:一般使用者等级的信息
        • LOG_UUCP:UUCP子系统
  • syslog()

    • 作用:向日志文件中写消息,可以规定消息的优先级、输出格式等
    • 原型:void syslog(int priority, char *format, ...);
    • 参数:
      • priority:指定信息等级
        • LOG_EMERG:系统无法继续使用
        • LOG_ALERT:需要立即采取措施
        • LOG_CRIT:有重要情况发送
        • LOG_ERR:有错误发生
        • LOG_WARNING:有警告发生
        • LOG_NOTICE:正常情况,但也是重要情况
        • LOG_INFO:信息消息
        • LOG_DEBUG:调试信息
      • format:类似printf中的字符格式
  • closelog()

    • 作用:关闭系统日志服务的连接
    • 原型:void closelog(void);

重写上节代码

#include <stdio.h>  
#include <stdlib.h>  
#include <string.h>  
#include <unistd.h>  
#include <sys/wait.h>  
#include <sys/types.h>  
#include <fcntl.h>  
  
int main() 
{    
    pid_t pid;  
    int i, fd, len;  
    char *buf = "守护进程运行中.\n";  
    len = strlen(buf)+1;
    
    pid = fork();	//1.1 创建子进程
    if (pid < 0) {  
    	printf("fork error!");  
    	exit(1);  
    }
    if (pid>0) 		// 1.2父进程退出  
    	exit(0);  
    
   /*打开系统日志服务连接*/
   openlog("daemon_syslog", LOG_PID, LOG_DAEMON);
    
   if ((sid = setsid()) < 0 ) 		// 2.在子进程中创建新会话。
   {
       syslog(LOG_ERR, "%s\n", "setsid");
       exit(1);
   }
    
   if ((sid = chdir("/")) < 0) 	// 3.设置工作目录为根目录  
   {
       syslog(LOG_ERR, "%s\n", "chdir");
       exit(1);
   }
    
    umask(0); 		// 4.设置权限掩码  
    
    for(i=0; i<getdtablesize(); i++) //5.关闭用不到的文件描述符  
    	close(i);
    
    //6.守护进程功能实现
    while(1) 			// 死循环表征它将一直运行
    {
        if((fd = open("/var/log/daemon.log",\
                      O_CREAT|O_WRONLY|O_APPEND,0600)) < 0)
    	{
            syslog(LOG_ERR, "open");
            exit(1);  
    	}  
    	write(fd, buf, len);  // 将buf写到fd中  
    	close(fd);  
    	sleep(10);  
    }  
    closelog();
    exit(1);  
}

3、利用现有程序伪装成守护进程

  • 意思就是希望某一个程序能够在当前会话被关闭后,照样能够运行,方法也很简单,就是利用nohup命令。

  • 例如,我想运行一个命令sleep 1000,正常情况下,该命令在终端执行后,如果关闭该终端,命令随之结束,而不会等到1000秒之后。

  • 通过执行:nohup sleep 1000 &命令后,查看sleep的运行情况如下:

    image-20221203214435510

  • 关闭当前会话窗口,再重新打开,执行ps axjf | grep sleep命令发现它任然在执行中。

    image-20221203214533220

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Leon_George

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值