软硬件环境
- Ubuntu 16.04 64bit
概述
在类UNIX
的多任务操作系统中,守护进程就是运行在后台的计算机程序,而不是运行在前台可以由用户直接控制。典型的守护进程一般都会在进程名后加上字母d
,表示daemon
,如常见的mysqld
、telnetd
、sshd
等。这些进程没有控制终端,不受用户登录、注销、登出的影响,它们一直在运行着。
创建守护进程的一般步骤是这样的,首先fork
出一个子进程,父进程退出,然后子进程调用setsid
创建新的会话,成为守护进程,接着关闭文件描述符,改变当前目录为根目录,重设文件权限掩码,忽略SIGCHLD
信号 ,最后还有必要通过日志系统来记录出错信息,方便日后查询。
最后来看一个实例,注释都写得比较清楚了,就不再赘述了。
编写守护进程的实例
/*daemon.c*/
#include <stdio.h>
#include <time.h>
#include <unistd.h>
#include <signal.h>
#include <sys/param.h>
#include <sys/types.h>
#include <sys/stat.h>
void initDaemon(void)
{
int pid;
int i;
/* terminate parent process */
if(pid=fork())
exit(0);
/* exit when fork failed */
else if(pid< 0)
exit(1);
/* continue only when it it the first child process
** run in a new session
*/
setsid();
/* seperate from terminator,kill the first child process */
if(pid=fork())
exit(0);
else if(pid< 0)
exit(1);
/* close all file descriptors */
for(i=0;i< NOFILE;++i)
close(i);
chdir("/home/djstava");
/* reset file umask */
umask(0);
return;
}
void signalUSR1(int signal)
{
/*handle SIGUSR1*/
FILE *fp;
time_t t;
if((fp=fopen("log","a")) >=0)
{
t=time(0);
fprintf(fp,"Received signal(%d),at %s\n", signal, asctime(localtime(&t)) );
fclose(fp);
}
}
int main(int argc, char ** argv)
{
FILE *fp;
time_t t;
initDaemon();
/* ignore signal */
signal(SIGCHLD, SIG_IGN);
/* handle SIGUSR1 */
signal(SIGUSR1, signalUSR1);
while(1)
{
sleep(60);
if((fp=fopen("log","a")) >=0)
{
t=time(0);
fprintf(fp,"djstava at %s\n",asctime(localtime(&t)) );
fclose(fp);
}
}
}
编译运行
执行以下命令编译并运行
gcc -o daemon daemon.c
./daemon
ps ax
sudo kill -s SIGUSR1 6942(这里的进程号每台机器不一样,根据实际情况更改)
cat /home/djstava/log
最后在文件/etc/rc.local
末尾添加如下语句,使daemon
程序在系统开机时就自动启动,最后需要将编译产生的daemon
文件copy
到/bin
目录下
su -djstava -c "/bin/daemon"
djstava
是我ubuntu
系统的用户名
最后附上一张UNIX
信号表,方便查阅