Linux--进程组、会话、守护进程

1:进程组 一个或多个进程的集合

     #include <unistd.h>
    pid_t getpgrp(void);
    getpgrp()用来取得目前进程所属的组识别码。
    此函数相当于调用getpgid(0)


     #include <unistd.h>
    pid_t getpgid(pid_t pid);
    getpgid()用来取得参数pid指定进程所属的组识别码。
    如果参数pid为0,则会取得目前进程的组识别码。执行成功则返回组识别码,如果有错误则返回-1。


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

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>

  4. int main() 
  5. {
  6.     pid_t pid;

  7.     if ((pid=fork())<0) 
  8.     {
  9.         printf("fork error!");
  10.     }
  11.     else if (pid==0) 
  12.     {
  13.         printf("The child process PID is %d.\n",getpid());
  14.         printf("The Group ID is %d.\n",getpgrp());
  15.         printf("The Group ID is %d.\n",getpgid(0));
  16.         printf("The Group ID is %d.\n",getpgid(getpid()));
  17.         exit(0);
  18.     }

  19.     sleep(3);
  20.     printf("The parent process PID is %d.\n",getpid());
  21.     printf("The Group ID is %d.\n",getpgrp());

  22.     return 0;
  23. }

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

组长进程
    组长进程标识: 其进程组ID==其进程ID
    组长进程可以创建一个进程组,创建该进程组中的进程,然后终止
    只要进程组中有一个进程存在,进程组就存在,与组长进程是否终止无关
    进程组生存期: 进程组创建到最后一个进程离开(终止或转移到另一个进程组)
  
    一个进程可以为
自己或子进程 设置进程组ID
    setpgid()加入一个现有的进程组或创建一个新进程组        
    
    #include <unistd.h>

    int setpgid(pid_t pid, pid_t pgid);
    
setpgid()将参数pid指定进程所属的组识别码设为参数pgid指定 的组识别码。
    如果参数pid 为0,则会用来设置目前进程的组识别
码,如果参数pgid为0,则会以目前进程的进程识别码来取代。

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

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>

  4. int main() 
  5. {
  6.     pid_t pid;

  7.     if ((pid=fork())<0) 
  8.     {
  9.         printf("fork error!");
  10.         exit(1);
  11.     }
  12.     else if (pid==0) 
  13.     {
  14.         printf("The child process PID is %d.\n",getpid());
  15.         printf("The Group ID of child is %d.\n",getpgid(0)); // 返回组id
  16.         sleep(5);
  17.         printf("The Group ID of child is changed to %d.\n",getpgid(0));
  18.         exit(0);
  19.     }

  20.     sleep(1);
  21.     setpgid(pid,pid); // 改变子进程的组id为子进程本身
  22.     
  23.     sleep(5);
  24.     printf("The parent process PID is %d.\n",getpid());
  25.     printf("The parent of parent process PID is %d.\n",getppid());
  26.     printf("The Group ID of parent is %d.\n",getpgid(0));
  27.     setpgid(getpid(),getppid()); // 改变父进程的组id为父进程的父进程
  28.     printf("The Group ID of parent is changed to %d.\n",getpgid(0));

  29.     return 0;
  30. }


2:会话一个或多个进程组的集合
    开始于用户登录
    终止与用户退出
    此期间所有进程都属于这个会话期


    建立新会话:setsid()函数
    该调用进程是组长进程,则出错返回
    先调用fork, 父进程终止,子进程调用
    该调用进程不是组长进程,则创建一个新会话
    ?该进程变成新会话首进程(session header)
    ?该进程成为一个新进程组的组长进程。
    ?该进程没有控制终端,如果之前有,则会被中断
    组长进程不能成为新会话首进程,新会话首进程必定会成为组长进程...

    #include <unistd.h>
    pid_t setsid(void);
    当进程是会话的领头进程时setsid()调用失败并返回(-1)。
    setsid()调用成功后,返回新的会话的ID,调用setsid函数的进程成为新的会话的领头进程,并与其父进程的会话组和进程组脱离。
    由于会话对控制终端的独占性,进程同时与控制终端脱离。

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

    
    #include <unistd.h>
    pid_t getsid(pid_t pid);
    获得会话首进程的进程id

   
若成功则返回进程组ID,若出错则返回-1     

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>

  4. int main() 
  5. {
  6.     pid_t pid;

  7.     if ((pid=fork())<0) 
  8.     {
  9.         printf("fork error!");
  10.         exit(1);
  11.     }
  12.     else if (pid==0) 
  13.     {
  14.         printf("The child process PID is %d.\n",getpid());
  15.         printf("The Group ID of child is %d.\n",getpgid(0));
  16.         printf("The Session ID of child is %d.\n",getsid(0));
  17.         sleep(10);
  18.         setsid(); // 子进程非组长进程,故其成为新会话首进程,且成为组长进程。该进程组id即为会话进程
  19.         printf("Changed:\n");
  20.         printf("The child process PID is %d.\n",getpid());
  21.         printf("The Group ID of child is %d.\n",getpgid(0));
  22.         printf("The Session ID of child is %d.\n",getsid(0));
  23.         sleep(20);
  24.         exit(0);
  25.     }

  26.     return 0;
  27. } 

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

3:守护进程
    Linux大多数服务都是通过守护进程实现的,完成许多系统任务
    0: 调度进程,称为交换进程(swapper),内核一部分,系统进程
    1: init进程, 内核调用,负责内核启动后启动Linux系统
    没有终端限制
    让某个进程不因为用户、终端或者其他的变化而受到影响,那么就必须把这个进程变成一个守护进程
 
守护进程编程步骤
    1. 创建子进程,父进程退出
        ? 所有工作在子进程中进行
        ? 形式上脱离了控制终端
    2. 在子进程中创建新会话
        ? setsid()函数
        ? 使子进程完全独立出来,脱离控制
    3. 改变当前目录为根目录
        ? chdir()函数
        ? 防止占用可卸载的文件系统
        ? 也可以换成其它路径
    4. 重设文件权限掩码
        ? umask()函数
        ? 防止继承的文件创建屏蔽字拒绝某些权限
        ? 增加守护进程灵活性
    5. 关闭文件描述符
        ? 继承的打开文件不会用到,浪费系统资源,无法卸载
        ? getdtablesize()
        ? 返回所在进程的文件描述符表的项数,即该进程打开的文 件数目

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <unistd.h>
  5. #include <sys/wait.h>
  6. #include <sys/types.h>
  7. #include <fcntl.h>

  8. int main() 
  9. {
  10.     pid_t pid;
  11.     int i,fd;
  12.     char *buf="This is a daemon program.\n";

  13.     if ((pid=fork())<0) 
  14.     {
  15.         printf("fork error!");
  16.         exit(1);
  17.     }
  18.     else if (pid>0) // fork且退出父进程
  19.         exit(0);
  20.     
  21.     setsid();     // 在子进程中创建新会话。
  22.     chdir("/");   // 设置工作目录为根
  23.     umask(0);     // 设置权限掩码
  24.     for(i=0;i<getdtablesize();i++) //getdtablesize返回子进程文件描述符表的项数
  25.         close(i);     // 关闭这些不将用到的文件描述符

  26.     while(1)      // 死循环表征它将一直运行
  27.     {
  28.         // 以读写方式打开"/tmp/daemon.log",返回的文件描述符赋给fd
  29.         if ((fd=open("/tmp/daemon.log",O_CREAT|O_WRONLY|O_APPEND,0600))<0) 
  30.         {
  31.             printf("Open file error!\n");
  32.             exit(1);
  33.         }
  34.         // 将buf写到fd中
  35.         write(fd,buf,strlen(buf)+1);
  36.         close(fd);
  37.         sleep(10);
  38.         printf("Never output!\n");
  39.     }

  40.     return 0;
  41. }

输出的内容存储到了日志文件中。。。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值