UNIX环境高级编程——进程关系
9.1 引言
本章详细说明进程组以及会话的概念,还将介绍登录shell(登录时所调用的)和所有从登录shell启动的进程之间的关系。
9.2 终端登录

9.3 网络登录

9.4 进程组
- 每个进程除了有一进程ID之外,还属于一个进程组,进程组是一个或多个进程的集合;
- 同一进程组中的各进程接收来自同一终端的各种信号;
- 每个进程组有一个唯一的进程组ID,它是一个正整数。
getpgrp函数返回调用进程的进程组ID:
#include <unistd.h>pid_t getpgrp(void);// 返回值:调用进程的进程组ID
getpgid函数返回指定进程的进程组ID:
#include <unistd.h>pid_t getpgid(pid_t pid);// 返回值:若成功,返回进程组ID;若出错,返回-1
- 若pid参数是0,返回调用进程的进程组ID,即getpgid(0)等效于个getpgrp()。
进程组相关概念:
- 每个进程组有一个组长进程,组长进程的进程组ID等于其进程ID;
- 进程组组长可以创建一个进程组、创建该组中的进程,然后终止;
- 只要在某个进程组中有一个进程存在,则该进程组就存在,这与其组长进程是否终止无关;
- 从进程组创建开始到其中最后一个进程离开为止的时间区间称为进程组的生命期;
- 某个进程组中的最后一个进程可以终止,也可以转移到另一个进程组。
进程调用setpgid可以加入一个现有的进程组或者创建一个新进程组:
#include <unistd.h>int setpgid(pid_t pid, pid_t pgid);// 返回值:若成功,返回0;若出错,返回-1
- 函数将pid进程的进程组ID设置为pgid;
- 如果两个参数相等,则由pid指定的进程变成组长进程;
- 如果pid是0,则使用调用者的进程ID;
- 如果pgid是0,则由pid指定的进程ID用作进程组ID;
- 一个进程只能为它自己或它的子进程设置进程组ID,且子进程调用了
exec后,它就不再更改该子进程的进程组ID。
9.5 会话
会话是一个或多个进程组的集合。

进程调用setsid函数建立一个新会话:
#include <unistd.h>pid_t setsid(void);// 返回值:若成功,返回进程组ID;若出错,返回-1
- 如果调用此函数的进程不是一个进程组的组长,则此函数创建一个新会话,具体会发生以下3件事:
(1)该进程变成新会话的会话首进程(会话首进程是创建该会话的进程),此时,该进程是新会话中的唯一进程,可将会话首进程ID视为会话ID;
(2)该进程成为一个新进程组的组长进程,新进程组ID是该调用进程的进程ID;
(3)该进程没有控制终端(新会话没有控制终端),如果在调用setsid之前该进程有一个控制终端,那么这种联系也被切断。 - 如果该调用进程已经是一个进程组的组长,则此函数返回出错。
getsid函数返回会话首进程的进程组ID:
#include <unistd.h>pid_t getsid(pid_t pid);// 返回值:若成功,返回会话首进程的进程组ID;若出错,返回-1
- 如若pid是0,
getsid返回调用进程的会话首进程的进程组ID; - 如若pid并不属于调用者所在的会话,那么调用进程就不能得到该会话首进程的进程组ID。
9.6 控制终端
会话和进程组还有一些其他特性:
- 一个会话可以有一个控制终端,这通常是终端设备(在终端登录情况下)或伪终端设备(在网络登录情况下);
- 建立与控制终端连接的会话首进程被称为控制进程;
- 一个会话中的几个进程组可被分为一个前台进程组以及一个或多个后台进程组;
- 如果一个会话有一个控制终端,则它有一个前台进程组,其他进程组为后台进程组;
- 无论何时键入终端的中断键(常常是Delete或Ctrl+C),都会将中断信号发送至前台进程组的所有进程;
- 无论何时键入终端的退出键(常常是Ctrl+\),都会将退出信号发送至前台进程组的所有进程;
- 如果终端接口检测到调制解调器(或网络)已经断开连接,则将挂断信号发送至控制进程(会话首进程)。

9.7 函数tcgetpgrp、tcsetpgrp和tcgetsid
需要有一种方法来通知内核哪一个进程组是前台进程组,这样,终端设备驱动程序就能知道将终端输入和终端产生的信号发送到何处:
#include <unistd.h>pid_t tcgetpgrp(int fd);// 返回值:若成功,返回前台进程组ID;若出错,返回-1
int tcsetpgrp(int fd, pid_t pgrpid);// 返回值:若成功,返回0;若出错,返回-1
- 函数
tcgetpgrp返回前台进程组ID,它与在fd上打开的终端相关联; - 如果进程有一个控制终端,则该进程可以调用
tcsetpgrp将前台进程组ID设置为pgrpid,pgrpid值应当是在同一会话中的一个进程组的ID,fd必须引用该会话的控制终端。
给出控制TTY的文件描述符,通过tcgetsid函数,应用程序就能获得会话首进程的进程组ID:
#include <termios.h>pid_t tcgetsid(int fd);// 返回值:若成功,返回会话首进程的进程组ID;若出错,返回-1
9.8 作业控制
3个特殊字符可使终端驱动程序产生信号,并将它们发送至前台进程组:
- 中断字符(一般采用Delete或Ctrl+C)产生SIGINT;
- 退出字符(一般采用Ctrl+\)产生SIGQUIT;
- 挂起字符(一般采用Ctrl+Z)产生SIGTSTP。

9.9 shell执行程序
shell处理管道的方式:
- shell
fork一个它自身的副本,然后此副本再为管道中的每条命令各fork一个进程; - 管道中的最后一个进程是shell的子进程,而执行管道中其他命令的进程则是该最后进程的子进程。

9.10 孤儿进程组
- 一个其父进程已终止的进程称为孤儿进程,这种进程由init进程“收养”;
- 整个进程组也可成为“孤儿”,孤儿进程组定义为:该组中每个成员的父进程要么是该组的一个成员,要么不是该组所属会话的成员。孤儿进程组的另一种描述:一个进程组不是孤儿进程组的条件是——该组中有一个进程,其父进程在属于同一会话的另一个组中。
- 如果进程组不是孤儿进程组,那么在属于同一会话的另一个组中的父进程就有机会重新启动该组中停止的进程。
若父进程是由shell作为前台作业执行的,当父进程终止时,子进程变成后台进程组。
9.11 FreeBSD实现

9.12 实例代码
chapter9
相关文章:
UNIX环境高级编程——进程关系
9.1 引言 本章详细说明进程组以及会话的概念,还将介绍登录shell(登录时所调用的)和所有从登录shell启动的进程之间的关系。 9.2 终端登录 9.3 网络登录 9.4 进程组 每个进程除了有一进程ID之外,还属于一个进程组,进…...
C# ref和out用法和区别
首先:两者都是按地址传递的,使用后都将改变原来参数的数值。 其次:ref可以把参数的数值传递进函数,但是out是要把参数清空,就是说你无法把一个数值从out传递进去的,out进去后,参数的数值为空&am…...
信息复制的革命:印刷术【提高信噪比】
文章目录 引言I 保证信息不被噪音所影响1.1 校对抄写错误的方法1.2 印刷术II 雕版印刷和活字印刷2.1 雕版印刷术2.2 毕昇的胶泥活字印刷2.3 古腾堡的铅活字印刷引言 科学的诞生,丰富了信息产生的源头。文字和纸张,加速了信息的传播和文明的进步。I 保证信息不被噪音所影响 复…...
【MySQL】事务
事务是一组操作的集合,我们将一组操作视为一个整体,所以事务里面的操作的时候要么同时成功,要么同时失败,之所以会有事务也是因为我们在实际生活中会用到 最典型的例子就是转账操作:A向B进行转账,A这边扣款成功的同时B那边一定是收款成功的,如果没有事务的话就会出现A扣款成功但…...
学习HCIP的day.03
目录 OSPF:开放式最短路径优先协议 OSPF的数据包 -- 5种 OSPF的状态机 OSPF的工作过程 OSPF的基础配置 关于OSPF协议从邻居建立成为邻接的条件 OSPF的接口网络类型 OSPF:开放式最短路径优先协议 无类别链路状态型IGP协议;由于其基于拓…...
Maven项目的配置
Maven是什么?它的作用是什么? Maven是一种开源的构建工具,它可以自动化构建、测试、部署和管理Java项目。它提供了一个中心化的构建过程,包括依赖管理、项目结构管理、插件管理等,使得开发人员更方便地维护和协作应用…...
Spring Boot使用(基础)
目录 1.Spring Boot是什么? 2.Spring Boot使用 2.1Spring目录介绍 2.2SpringBoot的使用 1.Spring Boot是什么? Spring Boot就是Spring脚手架,就是为了简化Spring开发而诞生的 Spring Boot的优点: 1.快速集成框架,提供了秒级继承各种框架,提供了启动添加依赖的功能 2.内…...
6WINDGate-overview
6WINDGate Overview Author:Once Day Date:2023年4月29日 本文是对6WIND官网文档的整理和翻译,仅供学习和研究之用,原始文章可参考下面文档: 6WINDGate Documentation - 6WIND6WINDGate Modules — 6WINDGate Modul…...
Java8新特性-流式操作
在Java8中提供了新特性—流式操作,通过流式操作可以帮助我们对数据更快速的进行一些过滤、排序、去重、最大、最小等等操作并且内置了并行流将流划分成多个线程进行并行执行,提供更高效、快速的执行能力。接下来我们一起看看Java8为我们新增了哪些便捷呢…...
Nautilus Chain Layer 3 圆桌会议圆满举办,超4.8K用户观看
在 4 月 21 日,Nautilus Chain 举办了以 “Layer 3 区块链的意义和发展以及Crypto的演变”为主题的线上圆桌会议,我们邀请了众多行业嘉宾包括 GitcoinDAO社区管理者Bob jiang、Whalers Community 发起者崔棉大师、Chatpuppy 联合创始人 古千峰、Whalers …...
本地elasticsearch中文分词器 ik分词器安装及使用
ElasticSearch 内置了分词器,如标准分词器、简单分词器、空白词器等。但这些分词器对我们最常使用的中文并不友好,不能按我们的语言习惯进行分词。 ik分词器就是一个标准的中文分词器。它可以根据定义的字典对域进行分词,并且支持用户配置自…...
Java 中的异常处理机制是什么?如何使用它来处理程序中的异常?(七)
Java 中的异常处理机制是一种重要的编程技术,它能够帮助程序员更好地管理程序中出现的异常情况。本文将详细介绍 Java 中的异常处理机制,并提供示例来说明如何使用异常处理机制来捕获和处理程序中的异常。 什么是异常? 在程序运行过程中&am…...
基于UDQ的并网单相逆变器控制【同步参考系下单相并网全桥正弦PWM逆变器闭环控制】(Simulink)
💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…...
JAVA开发——常用的注解
目录 spring spring MVC Spring Boot AOP MyBatis MyBatis-Plus JavaWeb开发 spring Autowired:自动安装,通过类匹配自动注册相应的Bean。 Component:将一个通用的 Java 类标记为 Bean,由 Spring 容器管理。 Controller&…...
【Java笔试强训 24】
🎉🎉🎉点进来你就是我的人了博主主页:🙈🙈🙈戳一戳,欢迎大佬指点! 欢迎志同道合的朋友一起加油喔🤺🤺🤺 目录 一、选择题 二、编程题 🔥年终奖 …...
SpringCloud详解
SpringCloud是一个基于SpringBoot的分布式系统开发框架,它能够帮助我们快速、稳定地构建分布式系统。本篇博客将对SpringCloud进行详细解析,介绍SpringCloud的主要组件和相关应用场景,同时提供代码示例以帮助读者更好地掌握SpringCloud的实际…...
如何保障网络安全
网络安全是一个涵盖范围广、深入浅出的话题。随着互联网在现代社会中扮演的重要角色日益突出,网络安全问题成为各个领域所关注的焦点。在此,我们将从以下几个方面来阐述网络安全的重要性,并讨论几种保障网络安全的方式。 一、网络安全的重要性…...
网络基础:socket套接字
文章目录 1. 前导知识1.1 源MAC地址和目的MAC地址1.2 源IP地址和目的IP地址1.3 MAC地址和IP地址的配合1.4 源端口号和目的端口号1.5 Socket1.6 UCP协议和TCP协议1.7 网络字节序高低位高低地址大端和小端网络字节序常用转换函数 2. socket 网络编程2.1 socket 常见接口创建套接字…...
程序员如何学好PHP?做好这五个方面就够了
今天我想和大家分享一下程序员的第一份工作对自己的意义以及影响。首先,我们都知道第一份工作很重要,因为它决定了你以后的职业生涯的方向。你的第一份工作做的什么方向,很可能就是你以后职业生涯中最主要的方向。对我个人而言,我…...
【开源项目】Build your own X 构建自己的项目
【开源项目】Build your own X 构建自己的项目 简介 Build your own X 是一个精心收集了大量资源的项目指南,可以通过从头开始重新创建我们最喜爱的技术来掌握编程。 项目地址: https://github.com/codecrafters-io/build-your-own-x这些项目里的资源…...
Bleeding Llama漏洞深度剖析:Ollama CVE-2026-7482让30万台AI服务器“内存裸奔“
你以为把大模型部署在本地就高枕无忧了?Cyera研究团队最新披露的"Bleeding Llama"漏洞(CVE-2026-7482)给所有人泼了一盆冷水。这个藏在Ollama量化管道里的堆越界读取缺陷,能让攻击者零认证、零交互,仅用三次…...
从Bode图到PI参数:基于开环传函特性的转速环整定实战解析
1. 转速环PI参数整定的核心逻辑 我第一次接触电机控制时,面对转速环PI参数整定这个"黑箱"完全无从下手。直到把Bode图和PI参数的关系捋清楚,才发现原来频域分析才是解开这个谜团的钥匙。转速环本质上是个闭环系统,但我们要整定PI参…...
SingleFile:为什么你需要的不仅是网页保存,而是数字记忆的永恒守护?
SingleFile:为什么你需要的不仅是网页保存,而是数字记忆的永恒守护? 【免费下载链接】SingleFile Web Extension for saving a faithful copy of a complete web page in a single HTML file 项目地址: https://gitcode.com/gh_mirrors/si/…...
初创团队如何利用 Taotoken 低成本启动 AI 功能开发与迭代
🚀 告别海外账号与网络限制!稳定直连全球优质大模型,限时半价接入中。 👉 点击领取海量免费额度 初创团队如何利用 Taotoken 低成本启动 AI 功能开发与迭代 对于资源有限的初创团队而言,在开发具备 AI 功能的产品时&a…...
对比直接使用厂商API,通过Taotoken调用大模型的延迟体感差异
🚀 告别海外账号与网络限制!稳定直连全球优质大模型,限时半价接入中。 👉 点击领取海量免费额度 对比直接使用厂商API,通过Taotoken调用大模型的延迟体感差异 1. 关于延迟体感的说明 在接入大模型服务时,…...
Gemini3.1Pro成本优化实战指南
在做 2026 年的多模态项目时,大家最关心的往往不是“能不能用”,而是“怎么用得更划算”。如果你正在对接不同模型、比较价格与可用性,先把计费规则与调用链路梳理清楚,会省下大量试错成本。你也可以把 KULAAI(dl.877a…...
实用开源工具:3步解决游戏按键冲突的SOCD清理最佳实践指南
实用开源工具:3步解决游戏按键冲突的SOCD清理最佳实践指南 【免费下载链接】socd Key remapper for epic gamers 项目地址: https://gitcode.com/gh_mirrors/so/socd 你是否曾在激烈的游戏对战中,明明按下了正确的方向键,角色却做出了…...
S型速度曲线进阶:基于Sin²(x)的PLC平滑运动控制实践(以伺服/步进系统为例)
1. 为什么伺服系统更需要S型速度曲线? 我第一次在CNC雕刻机上调试伺服电机时,发现用传统梯形速度曲线会导致雕刻边缘出现明显振纹。当时百思不得其解,直到用示波器捕捉到电机转矩的突变信号才恍然大悟——这就像开车时突然猛踩油门࿰…...
Navicat Premium 试用期重置实践:3种技术方案深度解析
Navicat Premium 试用期重置实践:3种技术方案深度解析 【免费下载链接】navicat_reset_mac navicat mac版无限重置试用期脚本 Navicat Mac Version Unlimited Trial Reset Script 项目地址: https://gitcode.com/gh_mirrors/na/navicat_reset_mac 对于macOS平…...
【技术底稿 31】Milvus 2.5.14 实战避坑实录:字段缺失、行数不匹配、Metadata JSON 类型三连坑完整解法
一、项目背景重构 RAG 底座、弃用 LangChain4j 后,改用 Milvus 原生 SDK 自研 Starter 做向量入库。自建文档分片、Ollama 嵌入向量生成,对接 Milvus 2.5.14 做向量持久化。过程中连续遇到三个经典致命报错:必填字段缺失、多字段行数不统一、…...
