守护进程

概念

守护进程(daemon)是一种运行在后台的特殊进程,它独立于控制终端并周期的执行某种任务或等待处理某些发生的事件。在Linux中,每个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程都会依附于这个终端,这个终端称为这些进程的控制终端,当控制终端被关闭的时候,相应的进程都会自动关闭。但是守护进程能突破这种限制,它脱离于终端在后台运行,并且它脱离终端的目的是为了避免进程在运行过程中的信息在任何终端中显示并且进程也不会被任何终端所产生的信息打断。它从被执行的时候开始运转,直到整个系统关闭才退出(当然可以认为杀死相应的守护进程)。如果想让某个进程不因为用户或中断或其他变化而改变,那么就必须把这个进程变成一个守护进程。


实现步骤

创建子进程

创建子进程,父进程退出(使子进程成为孤儿进程):这是编写守护进程的第一步,由于守护进程是脱离终端的,因此完成第一步后就会在shell终端里造成一个程序已经运行完毕的假象。之后的所有工作在子进程中完成,而用户在shell终端里则可以执行其他命令,从而在形式上做到了与控制终端脱离。

创建新会话

在子进程中创建新的会话(脱离控制终端):这步是创建守护进程中最重要的一步,虽然实现起来很简单,但是它的意义非常重要,在这里使用的是系统函数setsid()来创建一个新的会话,并且担任该会话组的组长。在这里有两个概念要解释一下,进程组和会话期:

进程组:是一个或多个进程的集合。进程组有进程组ID来唯一标识。除了进程号(PID)之外,进程组ID也是一个进程的必备属性。每个进程组都有一个组长进程,其组长进程的进程号等于进程组ID。且该进程组ID不会因组长进程的退出而受到影响。

会话周期:会话期是一个或多个进程的集合,通常一个会话开始于用户的登录,终止于用户的退出,在此期间用户运行的所有进程都属于这个会话期。

setsid()相关

setsid()作用:创建一个新的会话,并且担任该会话组的组长。具体作用包括:让一个进程摆脱原会话的控制,让进程摆脱原进程的控制,让进程摆脱原控制终端的控制。

创建守护进程要调用setsid()函数的原因:由于创建守护进程的第一步是调用fork函数来创建子进程,再将父进程退出。由于在调用fork()函数的时候,**子进程拷贝了父进程的会话期、进程组、控制终端等资源,虽然父进程退出了,但是会话期、进程组、控制终端都没有改变,因此,需要用setsid()函数将该子进程完全独立出来,从而摆脱其他进程的控制。

改变目录为根目录

使用fork()创建子进程继承了父进程的当前工作目录,由于在进程运行中,当前目录所在的文件系统是不能卸载的,这对以后使用会造成很多的麻烦。因此通常的做法是让“/”作为守护进程的当前目录,当然也可以指定其他的别的目录来作为守护进程的工作目录。

重设文件权限掩码

文件权限掩码是屏蔽掉文件权限中的对应位。由于使用fork()函数新创建的子进程继承了父进程的文件权限掩码,这就给该子进程使用文件带了很多的麻烦(比如父进程中的文件没有执行文件的权限,然而在子进程中希望执行相应的文件这个时候就会出问题)。因此在子进程中要把文件的权限掩码设置成为0,即在此时有最大的权限,这样可以大大增强该守护进程的灵活性。设置的方法是:umask(0)。

关闭文件描述符

同文件权限码一样,用fork() 函数新建的子进程会从父进程那里继承一些已经打开了的文件。这些文件被打开的文件可能永远不会被守护进程读写,如果不进行关闭的话将会浪费系统的资源,造成进程所在的文件系统无法卸下以及引起预料的错误。

守护进程的退出

当用户需要外部停止守护进程的运行时,往往需要使用kill命令来停止该守护进程,所以守护进程中需要编码来实现kill发出的signal 信号处理,达到进程的正常退出。
实现该过程的函数是signal函数,void sigterm_handler(int sig);

后台进程不等于守护进程

两者最重要的区别在于:守护进程没有控制终端,后台进程有。

  • 后台的文件描述符也是继承父进程,例如shell,所以它可以在当前终端下显示输出数据;
    但是daemon进程自己变成组长进程,其文件描述符号和控制终端没有关联,是控制台无关的;

  • 基本上任何一个程序都可以在后台运行,但守护进程是具有特殊要求的程序,比如要脱离自己的父进程,称为自己的会话组长等,这些要在代码中显式的写出来;

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值