守护进程

概念:

守护进程,也就是通常所说的Daemon进程,是Linux中的后台服务进程。周期性的执行某种任务或等待处理某些发生的事件。

Linux系统有很多守护进程,大多数服务都是用守护进程实现的。比如:像我们的tftp,samba,nfs等相关服务。

UNIX的守护进程一般都命名为*d的形式,如httpd,telnetd等等。

生命周期:

守护进程会长时间运行,常常在系统启动时就开始运行,直到系统关闭时才终止。

守护进程不依赖于终端

从终端开始运行的进程都会依附于这个终端,这个终端称为这些进程的控制终端。当控制终端被关闭时,相应的进程都会被自动关闭。咱们平常写进程时,一个死循环程序,咱们不知道有ctrl+c的时候,怎么关闭它呀,是不是关闭终端呀。也就是说关闭终端的同时也关闭了我们的程序,但是对于守护进程来说,其生命周期守护需要突破这种限制,它从开始运行,直到整个系统关闭才会退出,所以守护进程不能依赖于终端。

查看守护进程

ps axj a: 显示所有 x:显示没有控制终端的进程 j:显示与作业有关的信息(显示的列):会话期ID(SID),进程组ID(PGID),控制终端(TT),终端进程组ID(TRGID)

• 所有的守护进程都是以超级用户启动的(UID为0); • 没有控制终端(TTY为?); • 终端进程组ID为-1(TPGID表示终端进程组ID,该值表示与控制终端相关的前台进程组,如果未和任何终端相关,其值为-1; • 所有的守护进程的父进程:

历史上,Linux 的启动一直采用init进程;下面的命令用来启动服务。这种方法有两个缺点:1. 启动时间长。init进程是串行启动,只有前一个进程启动完,才会启动下一个进程。2. 启动脚本复杂。init进程只是执行启动脚本,不管其他事情。脚本需要自己处理各种情况,这往往使得脚本变得很长。

Systemd 就是为了解决这些问题而诞生的。它的设计目标是,为系统的启动和管理提供一套完整的解决方案。根据 Linux 惯例,字母d是守护进程(daemon)的缩写。 Systemd 这个名字的含义,就是它要守护整个系统。

进程组、会话、控制终端

• 进程组

shell里的每个进程都属于一个进程组,创建进程组的目的是用于简化向组内所有进程发送信号的操作,即如果一个信号是发给一个进程组,则这个组内的所有进程都会受到该信号【方便管理】。

• PGID进程组ID

进程组内的所有进程都有相同的PGID,等于该组组长的PID。(进程组组长:进程组中有一个进程担当组长。进程组ID(PGID)等于进程组组长的进程ID。已知一个进程,要得到该进程所属的进程组ID可以调用getpgrp。一个进程可以通过另一个系统调用setpgrp来加入一个已经存在的进程组或者创建一个新的进程组。

如果内核支持_POSIX_JOB_CONTROL(该宏被定义)则内核会为Shell 上的每一条命令行(可能由多个命令通过管道等连接)创建一个进程组。从这点上看,进程组不是进程的概念,而是shell上才有,所以在task_struct里并没有存储进程组id之类的变量。

进程组的生命周期到组中最后一个进程终止或其加入其他进程组(离开本进程组)为止。

会话

一般一个用户登录后新建一个会话,每个会话也有一个ID来标识(SID)。登录后的第一个进程叫做会话领头进程(session leader),通常是一个shell/bash。对于会话领头进程,其PID=SID。

控制终端

一个会话一般会拥有一个控制终端用于执行IO操作。会话的领头进程打开一个终端之后, 该终端就成为该会话的控制终端。与控制终端建立连接的会话领头进程也称为控制进程 (controlling process) 。一个会话只能有一个控制终端。

前台进程组

该进程组中的进程能够向终端设备进行读、写操作的进程组。例如登陆shell(例如bash)通过调用int tcsetpgrp(int fd, pid_t pgrp); 函数设置为某个进程组pgrp关联终端设备fd,该函数执行成功后,该进程组pgrp成为前台进程组。

后台进程组

该进程组中的进程只能够向终端设备写。

终端进程组ID

每个进程还有一个属性,终端进程组ID(TPGID),用来标识一个进程是否处于一个和终端相关的进程组中。前台进程组中的进程的TPGID=PGID,后台进程组的PGID≠TPGID。若该进程和任何终端无关,其值为-1。通过比较他们来判断一个进程是属于前台进程组,还是后台进程组。

进程组、对话期和控制终端关系

进程组、对话期和控制终端关系

  1. 每个会话有且只有一个前台进程组,但会有0个或者多个后台进程组。
  2. 产生在控制终端上的输入(Input)和信号(Signal)将发送给会话的前台进程组中的所有进程。对于输出(Output)来说,则是在前台和后台共享的,即前台和后台的打印输出都会显示在屏幕上。
  3. 终端上的连接断开时 (比如网络断开或 Modem 断开), 挂起信号将发送到控制进程(controlling process) 。
  4. 一个用户登录后创建一个会话。一个会话中只存在一个前台进程组,但可以存在多个后台进程组。第一次登陆后第一个创建的进程是shell,也就是会话的领头进程,该领头进程缺省处于一个前台进程组中并打开一个控制终端可以进行数据的读写。当在shell里运行一行命令后(不带&)创建一个新的进程组,命令行中如果有多个命令会创建多个进程,这些进程都处于该新建进程组中,shell将该新建的进程组设置为前台进程组并将自己暂时设置为后台进程组。

举例

  1. 打开第一个终端执行命令:
ping 127.0.0.1 -aq | grep icmp &  // 通过管道将两个命令串接起来ping –q不显示timeout信息,将其设置到后台并running
  1. 在第一个终端继续执行命令,在前台再新建一个进程组。【注意没有&】
ping 127.0.0.1 -aq | grep icmp //在前台再新建一个进程组,
  1. 开启第二个终端并运行
 ps axj | grep pts/0      即过滤只看pts/0里的会话 PPID   PID  PGID   SID TTY      TPGID STAT   UID   TIME COMMAND 2109  2111  2111  2111 pts/0     2538 Ss    1000   0:01 bash 2111  2503  2503  2111 pts/0     2538 S     1000   0:00 ping 127.0.0.1 -aq 2111  2504  2503  2111 pts/0     2538 S     1000   0:00 grep --color=auto icmp 2111  2538  2538  2111 pts/0     2538 S+    1000   0:00 ping 127.0.0.2 -aq 2111  2539  2538  2111 pts/0     2538 S+    1000   0:00 grep --color=auto timeo

• SID都是2111,说明大家都在一个Session里 • 有三个进程组PGID 2111,2503和2538。 我们可以看到用|连起来的ping和grep是在一个进程组里的。 • 2538这个进程组是一个前台的进程组,因为其PGID==TGPID, 2503这个进程组是一个后台进程组

  1. 在第一个终端中执行Ctrl+C
  2. 在第二个终端里继续ps axj | grep pts/0
 PPID   PID  PGID   SID TTY      TPGID STAT   UID   TIME COMMAND 2109  2111  2111  2111 pts/0     2111 Ss+   1000   0:01 bash 2111  2503  2503  2111 pts/0     2111 S     1000   0:00 ping 127.0.0.1 -aq 2111  2504  2503  2111 pts/0     2111 S     1000   0:00 grep --color=auto icmp

• 2538那个前台进程组的所有进程都消失了,说明信号会发给前台进程组的所有进程 • 2111,即bash所在的那个进程组成为了前台进程组。

守护进程创建流程

守护进程创建流程如下:

1. 创建子进程,父进程退出 2. 在子进程中创建新会话 3. 改变当前目录为根目录 4. 重设文件权限掩码 5. 关闭文件描述符 

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

由于守护进程是脱离控制终端的,因此,完成第一步后就会在shell终端里造成一程序已经运行完毕的假象。之后的所有后续工作都在子进程中完成,而用户在shell终端里则可以执行其他的命令,从而在形式上做到了与控制终端的脱离。

由于父进程已经先于子进程退出,会造成子进程没有父进程,从而变成一个孤儿进程。在Linux中,每当系统发现一个孤儿进程,就会自动由1号进程收养。原先的子进程就会变成init进程的子进程。

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

setsid()函数的作用。一个进程调用setsid()函数后,会发生如下事件:

• 首先内核会创建一个新的会话,并让该进程成为该会话的leader进程,• 同时伴随该session的建立,一个新的进程组也会被创建,同时该进程成为该进程组的组长。• 该进程此时还没有和任何控制终端关联。若需要则要另外调用tcsetpgrp,前面讲前台进程组时介绍过。

调用setsid()有以下3个作用:

• 让进程摆脱原会话的控制。• 让进程摆脱原进程组的控制。• 让进程摆脱原控制终端的控制。

那么,在创建守护进程时为什么要调用setsid()函数呢?

读者可以回忆一下创建守护进程的第一步,在那里调用了fork()函数来创建子进程再令父进程退出。由于在调用fork()函数时,子进程全盘复制了父进程的会话期、进程组和控制终端等,虽然父进程退出了,但原先的会话期、进程组和控制终端等并没有改变,因此,还不是真正意义上的独立。而setsid()函数能够使进程完全独立出来,从而脱离所有其他进程和终端的控制。

详细见man 2 setsid。

3.改变当前目录为根目

这一步也是必要的步骤。使用fork()创建的子进程继承了父进程的当前工作目录。

由于在进程运行过程中,当前目录所在的文件系统(如“/mnt/usb”等)是不能卸载的,这对以后的使用会造成诸多的麻烦(如系统由于某种原因要进入单用户模式)。

因此,通常的做法是让“/”作为守护进程的当前工作目录,这样就可以避免上述问题。当然,如有特殊需要,也可以把当前工作目录换成其他的路径,如/tmp。改变工作目录的常见函数是chdir()。

4. 重设文件权限掩码

文件权限掩码是指屏蔽掉文件权限中的对应位。

例如,有一个文件权限掩码是050,它就屏蔽了文件组拥有者的可读与可执行权限。由于使用fork()函数新建的子进程继承了父进程的文件权限掩码,这就给该子进程使用文件带来了诸多的麻烦。

因此,把文件权限掩码设置为0,可以大大增强该守护进程的灵活性。设置文件权限掩码的函数是umask()。在这里,通常的使用方法为umask(0)。即赋予最大的能力。

5. 关闭文件描述符

同文件权限掩码一样,用fork()函数新建的子进程会从父进程那里继承一些已经打开的文件。这些被打开的文件可能永远不会被守护进程读或写,但它们一样消耗系统资源,而且可能导致所在的文件系统无法被卸载。

在上面的第(2)步之后,守护进程已经与所属的控制终端失去了联系,因此,从终端输入的字符不可能达到守护进程,守护进程中用常规方法(如printf())输出的字符也不可能在终端上显示出来。

所以,文件描述符为0、1和2的3个文件(常说的输入、输出和报错这3个文件)已经失去了存在的价值,也应被关闭。

代码实现

/* 关注一口Linux*/#include  #include  #include  #include #include #include  #include #include int main(){ pid_t pid; int i, fd; char *buf = "This is a Daemon"; pid = fork(); if (pid  0) {  exit(0);  } /* 第二步 */ setsid(); /* 第三步 */   chdir("/");   /* 第四步 */ umask(0); /* 第五步 */   for(i = 0; i 

执行结果

由上图可见: • 守护进程./run 的UID为0; • 没有控制终端(TTY为?); • 终端进程组ID为-1; • 守护进程的父进程为1516,即systemd。

更多Linux干货,请关注: 一口Linux

potplayer 多个进程_进程组、会话、控制终端概念,如何创建守护进程?相关推荐

  1. ora03135连接失去联系 进程id 0_进程组、会话、控制终端概念,如何创建守护进程?...

    守护进程 概念: 守护进程,也就是通常所说的Daemon进程,是Linux中的后台服务进程.周期性的执行某种任务或等待处理某些发生的事件. Linux系统有很多守护进程,大多数服务都是用守护进程实现的 ...

  2. potplayer 多个进程_创建守护进程的步骤

    什么是守护进程? 答:守护进程是后台运行的.系统启动是就存在的.不予任何终端关联的,用于处理一些系统级别任务的特殊进程. 实现思路: 实现一个守护进程,其实就是将普通进程按照上述特性改造为守护进程的过 ...

  3. 从进程组、会话、终端的概念深入理解守护进程

    从进程组.会话.终端的概念深入理解守护进程 一.写在前面 「守护进程」是 Linux 的一种长期运行的后台服务进程,也有人称它为「精灵进程」.我们常见的 httpd.named.sshd 等服务都是以 ...

  4. Linux系统编程---11(会话,守护进程,创建守护进程)

    会话 创建会话 创建一个会话需要注意以下6点注意事项 调用进程不能是进程组组长,该进程变成新会话首进程 该进程成为一个新进程组的组长进程 需要root权限(nbuntu不需要) 新会话丢弃原有的控制终 ...

  5. Linux创建守护进程

    守护进程(Daemon)是运行在后台的一种特殊进程.它独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件.它不需要用户输入就能运行而且提供某种服务,不是对整个系统就是对某个用户程序提供服务 ...

  6. linux之创建守护进程

    简介 守护进程是一种运行在后台的特殊进程,它独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件. 其实,本质上守护进程和普通的进程并没有什么区别,只是我们规定了一种进程的编写规则,将其叫做 ...

  7. 守护进程与后台进程(Python 创建守护进程)

    文章目录 一.守护进程与后台进程 1. 守护进程 1.1 代码实现 为什么要fork两次 umask权限掩码 进程组 会话组 2. 后台进程 3. 守护进程与后台进程区别 4. 使用场景总结 二.参考 ...

  8. 黑马程序员Linux系统开发视频之创建守护进程模型

    黑马程序员Linux系统开发视频之创建守护进程模型 1.创建子进程,父进程退出   所有工作在子进程中进行形式上脱离了控制终端 2.在子进程中创建新会话   setsid()函数   使子进程完全独立 ...

  9. python 守护程序检测进程是否存在_python创建守护进程的疑问

    我自己写了一个简易的下载和文件执行的客户端,如下 """ 省略若干代码 """ #执行下载函数 def do_script(): " ...

最新文章

  1. 论文: Generating Software Test Data by Evolution---自己觉得很好的句子----(1)introduction
  2. mysql show语句_mysql常用show语句
  3. NetBeans 6.5 正式发布
  4. ubuntu11.10 源码编译安装php5.3.8,Ubuntu 11.10编译安装Nginx、PHP 5.3.8、MySQL、MongoDB、Memcached、SSL、SMTP...
  5. mysql索引动态维护_MySQL之——索引
  6. FISCO BCOS 智能合约 可视化数据库 Mysql MariaDB
  7. 常见视频编码格式解析
  8. 不知道Word转图片PDF怎么转?1分钟帮你快速转换
  9. 学习Vue3 第七章(认识Reactive全家桶)
  10. mysql 中的any_value 函数
  11. Thingworx配置mysql的jdbc
  12. 新浪微创投狩猎季:5家LP皆VC大佬 人气日渐高涨
  13. sap采购申请自动转采购订单_我的SAP运维日常_0021_MM_计划订单转采购申请时绑定凭证类型...
  14. Arcgis使用教程(十)ARCGIS地图制图之经纬网格添加
  15. 【SHOI 2002】百事世界杯之旅
  16. CentOS7下的网络设定方法
  17. 【软件构造】课件精译(三)软件生命周期与配置管理
  18. 计算机应用技术第二版笔记,(计算机应用技术笔记.doc
  19. MACD指标与成交量指标怎么配合使用?耳闻过成交量指标可以辅助MACD指标判势吗?
  20. 关闭research_Microsoft Research的新数码照片技术已集成到Microsoft Digital Image Suite 9中...

热门文章

  1. php怎么引用表单元素,表单元素:最全的各种html表单元素获取和使用方法总结...
  2. linux sudo 版本,Linu下如何升级当前sudo版本
  3. 软件工程 质量问题_软件质量| 软件工程
  4. 非导向传输媒体| 计算机网络
  5. Java中常用的4个Json库,哪个性能更牛逼?
  6. Java 200+ 面试题补充 ThreadLocal 模块
  7. OpenGL 自定义着色器(Shaders)
  8. OpenGL Sharders(着色器) 入门
  9. 数据封装以及解封的过程
  10. 计算机数据库原理试题,计算机科学与技术专业《数据库原理》模拟试题(B)