当前位置: 首页 > news >正文

Linux——守护进程化(独立于用户会话的进程)

目录

前言

一、进程组ID与会话ID

二、setsid()  创建新会话

三、daemon 守护进程


前言

在之前,我们学习过socket编程中的udp通信与tcp通信,但是当时我们服务器启动的时候,都是以前台进程的方式启动的,这样很不优雅,因为前台进程无法接受命令输入,同时也可能一不小心被使用者终止。因此我们得让服务器以守护进程的方式进行运行。如果不太了解,可以看这篇文章前台进程与后台进程。

一、进程组ID与会话ID

我们输入 指令 ps axj 可以找到当前系统中的所有进程,并且查看如下各种信息。

  • PPID: 父进程ID。
  • PID: 进程ID。
  • PGID: 进程组ID。
  • SID: 会话ID。
  • TTY: 与进程关联的终端类型(如果有的话)。
  • TPGID: 终端的前台进程组ID(如果有的话)。
  • STAT: 进程状态(例如,S表示休眠,R表示运行等)。
  • UID: 用户ID,表示运行该进程的用户。
  • TIME: CPU时间,表示进程使用的CPU时间总量。
  • COMMAND: 启动进程的命令名或命令行。

我们主要来看进程组ID PGID 和会话ID SID。

如下,启动一个sleep进程,查询sleep进程的信息。发现sleep进程的组 PGID 就是自己的PID,PPID 就是bash进程的ID。也就是说该sleep自成进程组。 同时进程组默认是在一个会话中的。

如果我们又打开一个bash,一个命令创建三个sleep,会发现他们的PGID都是第一个创建的进程PID,同时SID都是bash。

他们的关系如下图所示。任何时刻,一个会话可以有很多进程组,bash是一个组,sleep1000 | sleep 2000 | sleep 3000 又是一个进程组,sleep 10000也是一个进程组。但默认情况只允许一个进程组在前台。

也就是说bash在前台的话, 命令行就能接受命令,如果前台被其他进程占据,那么bash就会退回到后台,也就无法接受命令了

因此,如果我们启动的服务器程序,是在当前会话中,那么当bash退出(shell关闭),或者被不小心kill 掉,那么我们的服务器也会随之崩溃。

如果想让服务器不受xshell用户登录和退出的影响,就要让服务器程序自成一个会话,不隶属于任何一个bash。这就是守护进程

二、setsid()  创建新会话

setsid() 可以创建一个新会话,调用之后,当前进程会与原会话和进程组脱离关系。这意味着进程不再属于原来的会话和进程组,而是成为新会话的领头进程,并可能创建一个新的进程组。

但是他有一个前提,调用setsid的进程不能是进程组的组长

也就是说之前的sleep 1000 | sleep 2000 | sleep 30000 中,sleep 1000进程是进程组的组长,那么他就无法使用setsid。

同时,如果当前进程组只有一个进程,那就默认该进程就是组长,也无法使用setsid。

那么我们处理方式也很简单,让进程fork一下,使用子进程去setsid即可。也就是说守护进程一定就是孤儿进程

三、daemon 守护进程

对于具体的进程守护化,我们通常需要做如下内容

  1. 忽略可能一起进程异常退出的信号
  2. fork进程,让子进程去调用setsid()
  3. 改变cwd(当前工作目录),可选可不选,如果让cwd变为根目录,想误删也就不太可能,同时访问速度也会更快一点。
  4. 处理用户的标准输入、标准输出、标准错误。守护进程化后,通常不需要直接与用户进行交互,避免资源浪费和潜在的问题。
#pragma once#include <iostream>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <signal.h>//该文件相当于垃圾箱,往里面cout、cerr就相当于丢弃数据,cin相当于直接读到文件末尾
const char *dev_null = "/dev/null";void deamon(bool ischidr, bool isclose)
{// 1.忽略可能引起进程异常退出的信号signal(SIGCHLD, SIG_IGN);signal(SIGPIPE, SIG_IGN);// 2.fork进程,让子进程去setsidif (fork() > 0)exit(0);setsid();// 3.改变当前工作目录if (ischidr)chdir("/");// 4.处理用户的输入输出与错误int fd = open(dev_null, O_RDWR);if (isclose){//直接关闭close(0);close(1);close(2);}else{if (fd > 0){dup2(fd, 1);dup2(fd, 2);dup2(fd, 3);close(fd);}}
}

 库里面也有daemon函数,我们也是模仿库里面写的,但是自己写的daemon可控性会更高一些。

那么现在就可以对服务器(之前写的Tcp服务器)进程处理,直接调用deamon,同时日志的作用也体现出来了,将需要的消息放到日志中。

这样也能正常运行,并将进程守护化了,关闭shell也无法将线程退出,需要关闭机器或者使用kill命令。

 代码链接

http://www.lryc.cn/news/344482.html

相关文章:

  • 安卓开发--按键跳转页面,按键按下变色
  • Ps基础学习笔记
  • spring开发问题总结(持续更新)
  • Android 状态栏WiFi图标的显示逻辑
  • 更改 DeepXDE 的后端
  • SpringBoot之Zuul服务
  • Go-变量
  • 【CTF-Crypto】RSA-选择明密文攻击 一文通
  • Pytorch基础:torch.expand() 和 torch.repeat()
  • 如何正确安装Scrapy 2.6.1并解决常见的Python环境问题
  • 阵痛中的乳业产业,何时才能成为下一个啤酒产业?
  • 关于模型参数融合的思考
  • Windows MySQL本地服务器设置并导入数据库和数据
  • 豪投巨资,澳大利亚在追逐海市蜃楼吗?
  • 面试集中营—Redis架构篇
  • 05_kafka-整合springboot
  • 论UML在学情精准测评系统中的应用
  • Day23 代码随想录打卡|字符串篇---重复的子字符串
  • 【win10 文件夹数量和看到不一致查看隐藏文件已经打开,Thumb文件作妖】
  • ctfshow web入门 sql注入 web224--web233
  • 「Java开发指南」如何用MyEclipse搭建GWT 2.1和Spring?(一)
  • python同时进行字符串的多种替换
  • 【Java基础题型】用筛法求之N内的素数(老题型)
  • Linux进程——Linux环境变量
  • SRM系统供应链库存协同提升企业服务水平
  • Windows安全加固-账号与口令管理
  • 【数据库原理及应用】期末复习汇总高校期末真题试卷03
  • 数据库加密数据模糊匹配查询技术方案
  • jsSPA应用如何实现动态内容更新
  • C++学习笔记——仿函数