Linux守护进程基础

来源:岁月联盟 编辑:exp 时间:2011-09-14

1 守护进程中涉及到的基本概念


1.1进程组
1.1.1 进程组基本概念
进程组是一个或多个进程的集合,可以接收来自同一个终端的各种信号。每运行一个程序或是命令都将产生一个进程组。

每个进程属于一个进程组,而每个进程组都存在一个领头进程(或是叫组长进程),一般进程组的第一个进程是领头进程。领头进程可以创建一个进程组、创建该组中的进程。领头进程fork出的子进程也将在该进程组中,一旦子进程执行exec等退出函数就不再属于该进程组。

 


进程组的生命周期:从创建开始到最后一个进程离开为止成为进程组的生命周期。组中最后一个进程的离开可以是终止,或加入其他进程。进程组的生命周期与组长进程是否终止无关,只要有一个进程存在,进程组的生命周期就未结束。

1.1.2 相关命令
1、查看所有进程组pid(即领头进程的pid):ps –A –o pgrp= | sort |uniq

2、查看某个进程的进程组pid:ps –C 进程名 –o pgrp=

3、根据进程名获得进程pid:pidof 进程名

4、根据进程pid获得进程名:ps –aux |grep 进程pid

5、获取某个进程的具体信息:ps –ef |grep 进程名或进程pid

1.1.3 相关函数
#include <unistd.h>

pid_t getpgrp();    //获取进程组pid,即进组中的领头进程的pid,相当于调用getpgid(0)

pid_t getpid();     //获取当前进程pid

pid_t getppid();    //获取当前进程的父pid

pid_t getpgid();    //返回指定进程的进程组pid

int setpgid(pid_t pid, pid_t pgid); /*将pid进程的进程组id设置为pgid,如果两个参数相等,

                                                            *则pid指定的进程变成进程组组长。如果pid是0,则进程

                                                            *组id为pgid,如果pgid是0,则进程组id为pid。*/

int setpgrp();  //将当前进程所属的进程组的pid设置为当前进程的pid,相当于setpgid(0, 0)

1.2会话
1.2.1 会话基本概念
        一次登录形成一个会话,一个会话可包括多个进程组(www.linuxidc.com一个前台进程组和多个后台进程组),但只能有一个前台进程组。

1.2.2 相关函数
#include <unistd.h>

pid_t setsid();  

         当调用该函数的进程不是进程组的领头进程时,该函数才能建立新的会话。函数调用成功后,调用进程成为新会话的领头进程和新进程的领头进程,同时进程失去控制终端。

1.3控制终端
         进程组中有领头进程,类似地,会话中也对应有领头进程。会话的领头进程打开一个终端之后,该终端就成为会话的控制终端,与控制终端建立连接的会话的领头进程成为控制进程(session leader)。一个会话只能有一个控制终端,同时一个控制终端只能一个会话。         产生在控制端上的输入和信号(比如按下ctrl+c就会产生SIGINT信号)将发送给会话的前台进程组中的所有进程。

2守护进程及其特性
1、守护进程最重要的特性是后台运行;

2、守护进程必须与其运行前的环境脱离开来。这些环境包括未关闭的文件描述符,控制终端,会话和进程组,工作目录等,这些环境通常是从执行它的父进程(特别是Shell)中继承下来的。

3、守护进程可以在Linux系统启动时从脚本/etc/rc.d中启动,可以由作业规划进程crond启动,还可以由用户终端(通常是Shell)执行。

3 守护进程编程
        针对守护进程的特性可以将普通进程改造为守护进程。

3.1后台运行特性
让daemon在子进程中运行。

if( pid=fork() )

{

         exit(0);  //父进程结束,子进程继续

}

3.2脱离环境
3.2.1脱离进程组、登录会话和控制终端
         进程组、登录会话和控制终端通常都是从父进程继承下来的,为了不受它们的影响,控制终端必须将其摆脱。具体方法是调用setsid函数。

         通过if( pid=fork() ){exit(0);}将产生了子进程,能保证调用setsid的进程不是进程组中的领头进程。当setsid调用成功后,该子进程成为新会话中的领头进程和进程组的领头进程,并脱离了原来的会话、进程组和控制终端。

3.2.2禁止进程打开控制终端
if( pid=fork() )

{

         exit(0);  //该子进程结束,又产生子进程

}

         结束该子进程,产生新的子进程,此时的新子进程不是会话领头进程,所以不能打开控制终端。通过这种方式就能禁止进程打开控制终端。

3.2.3脱离打开的文件描述符
         进程从其父进程继承了打开的文件描述符。如不关闭,将会造成系统资源的浪费,同时,造成进程所在的文件系统无法卸载以及引起其他无法预料的错误。

max_fd = sysconf(_SC_OPEN_MAX);

for(i = 0; i < max_fd; i++)

{

close(i);

}


3.2.4脱离当前工作目录
进程活动时,其工作目录所在的文件系统不能卸载,一般需要将工作目录切换到根目录。对于需要运行日志的进程将工作目录改变到特定的目录。

chdir(“/”);

3.2.5重设文件的权限掩码 

         进程从其父进程继承的文件的权限掩码,可能会修改守护进程所创建的文件权限,需要将掩码清楚。

umask(0);

3.2.6处理SIGCHLD信号
         处理SIGCHLD信号并不是必须的。但对于某些进程,www.linuxidc.com 特别是服务器进程往往在请求到来时生成子进程处理请求。如果父进程没有等待(没有调用wait或waitpid)子进程的结束,子进程将成为僵尸进程(Zombie Process)从而占用系统资源。而如果父进程等待子进程结束,又将增加父进程的负担,影响服务器进程的并发性能。在Linux可以将SIGCHLD设置为SIG_IGN,这样内核在子进程结束时不会产生僵尸进程。

signal(SIGCHLD, SIG_IGN);  //屏蔽SIGCHLD信号

         这是常用于并发服务器提升性能的一个技巧。当服务器进程没有调用wait去清理子进程而产生僵尸子进程时,如果屏蔽了SIGCHLD信号,内核会把僵尸子进程转交给init进程处理。

4 守护进程示例
#include <unistd.h>    
#include <signal.h>    
#include <sys/types.h>    
#include <sys/stat.h>    
#include <stdio.h>    
#include <stdlib.h>   
  
void init_daemon()   
{   
  
         int pid;   
         int i;   
         int max_fd;  
  
   
         if(pid = fork())  
         {  
             exit(0); //父进程结束,子进程继续   
           }   
         else if(pid < 0)  
         {  
             exit(1); //fork失败              
           }  
  
         setsid();    
  
         if(pid = fork())  
         {  
             exit(0); //该子进程结束,又产生新子进程    
           }  
         else if(pid < 0)  
         {  
             exit(1); //fork错误,退出   
           }   
         
         /* 关闭打开的文件描述符*/   
         max_fd = sysconf(_SC_OPEN_MAX);  
  
         for(i = 0; i < max_fd; i++)  
         {   
             close(i);   
         }   
  
         /* 切换工作目录 */   
         chdir("/tmp");   
   
  
         /* 重设文件创建掩码 */   
         umask(0);   
  
         return;   
  
}   
  
int main()   
{   
  
         FILE *fp = NULL;   
  
         signal(SIGCHLD, SIG_IGN);   
  
         init_daemon();   
  
         while(1)   
         {   
  
             sleep(1);  
             if((fp = fopen("/mnt/hgfs/Share/unix/test.log", "a")) != NULL)  
             {   
                 fprintf(fp, "%s/n", "test message");   
                 fclose(fp);   
             }   
         }   
  
         return 0;   
}  
编译:gcc –o daemon daemon.c

运行:./daemon

查看守护进程:ps –ef |grep daemon

 /

查看该守护程序产生的文件:

 /

作者“Orion的博客”