进程组

  一个或多个进程的集合

  进程组ID: 正整数

  两个函数

  getpgid(0)=getpgrp()

有关linux--进程组、会话、守护进程详解_进程组

eg:显示子进程与父进程的进程组id

有关linux--进程组、会话、守护进程详解_进程组_02

 #include <stdio.h>
 #include <stdlib.h>
 #include <unistd.h>
 
  main() {
     pid_t pid;
 
      ((pid=fork())<) {
         printf();
     }  (pid==) {
         printf(,getpid());
         printf(,getpgrp());
         printf(,getpgid());
         printf(,getpgid(getpid()));
         exit();
     }
 
     sleep();
     printf(,getpid());
     printf(,getpgrp());
 
      ;
 }

有关linux--进程组、会话、守护进程详解_进程组_02

有关linux--进程组、会话、守护进程详解_linux/UNIX_04

进程组id = 父进程id,即父进程为组长进程

 

组长进程

  组长进程标识: 其进程组ID==其进程ID

  组长进程可以创建一个进程组,创建该进程组中的进程,然后终止

  只要进程组中有一个进程存在,进程组就存在,与组长进程是否终止无关

  进程组生存期: 进程组创建到最后一个进程离开(终止或转移到另一个进程组)

 

一个进程可以为自己或子进程设置进程组ID

  setpgid()加入一个现有的进程组或创建一个新进程组

有关linux--进程组、会话、守护进程详解_linux/UNIX_05

eg:父进程改变自身和子进程的组id

有关linux--进程组、会话、守护进程详解_进程组_02

 #include <stdio.h>
 #include <stdlib.h>
 #include <unistd.h>
 
  main() {
     pid_t pid;
 
      ((pid=fork())<) {
         printf();
         exit();
     }  (pid==) {
         printf(,getpid());
         printf(,getpgid());          sleep();
         printf(,getpgid());
         exit();
     }
 
     sleep();
     setpgid(pid,pid);      
     sleep();
     printf(,getpid());
     printf(,getppid());
     printf(,getpgid());
     setpgid(getpid(),getppid());      printf(,getpgid());
 
      ;
 }

有关linux--进程组、会话、守护进程详解_进程组_02

有关linux--进程组、会话、守护进程详解_进程组_08

 

会话: 一个或多个进程组的集合

  开始于用户登录

  终止与用户退出

  此期间所有进程都属于这个会话期

有关linux--进程组、会话、守护进程详解_进程组_09

建立新会话:setsid()函数

  该调用进程是组长进程,则出错返回

    先调用fork, 父进程终止,子进程调用

  该调用进程不是组长进程,则创建一个新会话

    该进程变成新会话首进程(session header)

    该进程成为一个新进程组的组长进程。

    该进程没有控制终端,如果之前有,则会被中断

组长进程不能成为新会话首进程,新会话首进程必定会成为组长进程...

有关linux--进程组、会话、守护进程详解_进程组_10

会话ID:会话首进程的进程组ID

获取会话ID: getsid()函数

有关linux--进程组、会话、守护进程详解_守护进程_11

有关linux--进程组、会话、守护进程详解_进程组_02

 #include <stdio.h>
 #include <stdlib.h>
 #include <unistd.h>
 
  main() {
     pid_t pid;
 
      ((pid=fork())<) {
         printf();
         exit();
     }  (pid==) {
         printf(,getpid());
         printf(,getpgid());
         printf(,getsid());
         sleep();
         setsid();          printf();
         printf(,getpid());
         printf(,getpgid());
         printf(,getsid());
         sleep();
         exit();
     }
 
      ;
 }

有关linux--进程组、会话、守护进程详解_进程组_02

 

有关linux--进程组、会话、守护进程详解_setpgid_14

在子进程中调用setsid()后,子进程成为新会话首进程,且成为一个组长进程,其进程组id等于会话id

 

 

守护进程

  Linux大多数服务都是通过守护进程实现的,完成许多系统任务

  0: 调度进程,称为交换进程(swapper),内核一部分,系统进程

  1: init进程, 内核调用,负责内核启动后启动Linux系统

  没有终端限制

  让某个进程不因为用户、终端或者其他的变化而受到影响,那么就必须把这个进程变成一个守护进程

 

守护进程编程步骤

  1. 创建子进程,父进程退出

    所有工作在子进程中进行

    形式上脱离了控制终端

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

    setsid()函数

    使子进程完全独立出来,脱离控制

  3. 改变当前目录为根目录

    chdir()函数

    防止占用可卸载的文件系统

    也可以换成其它路径

  4. 重设文件权限掩码

    umask()函数

    防止继承的文件创建屏蔽字拒绝某些权限

    增加守护进程灵活性

  5. 关闭文件描述符

    继承的打开文件不会用到,浪费系统资源,无法卸载

    getdtablesize()

    返回所在进程的文件描述符表的项数,即该进程打开的文件数目

有关linux--进程组、会话、守护进程详解_守护进程_15

有关linux--进程组、会话、守护进程详解_进程组_02

 #include <stdio.h>
 #include <stdlib.h>
 #include <.h>
 #include <unistd.h>
 #include <sys/wait.h>
 #include <sys/types.h>
 #include <fcntl.h>
 
  main() {
     pid_t pid;
      i,fd;
      *buf=;
 
      ((pid=fork())<) {
         printf();
         exit();
     }  (pid>)           exit();
     
     setsid();         chdir();       umask();         (i=;i<getdtablesize();i++)           close(i);                 
     () {           ((fd=open(,O_CREAT|O_WRONLY|O_APPEND,))<) {
             printf();
             exit();
         }
                  write(fd,buf,strlen(buf)+);
         close(fd);
         sleep();
         printf();
     }
 
      ;
 }

有关linux--进程组、会话、守护进程详解_进程组_02

因为stdout被关掉了,所以“Never ouput!”不会输出。

查看/tmp/daemon.log,说明该程序一直在运行

有关linux--进程组、会话、守护进程详解_守护进程_18


本文转自:http://www.cnblogs.com/forstudy/archive/2012/04/03/2427683.html