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

Linux 进程通信:信号与共享内存详解

一、信号通信

信号是 Linux 中用于异步通信、通知机制、处理随机事件的轻量级 IPC 方式,比如进程终止、异常通知等。

1. 信号的发送与接收流程

  1. 触发信号:根据需求(如用户按下 Ctrl+C)触发信号;
  2. 内核查找进程:Linux 接收信号请求,在 PCB(进程控制块)链表中找到目标进程的 PID;
  3. 中断进程执行信号处理函数:目标进程暂停当前工作,执行 PCB 中注册的信号处理函数(如handle2);
  4. 恢复原进程执行:信号处理函数执行完毕后,原进程继续运行。

2. 常见信号及默认行为

Linux 系统定义了多个信号,每个信号有其默认处理动作(可通过man 7 signal查看)。常见信号及默认行为如下:

信号名取值动作说明
SIGHUP1Term控制终端挂起
SIGINT2Term键盘中断(Ctrl+C)
SIGQUIT3Core键盘退出(Ctrl+\),并生成 core dump
SIGILL4Core非法指令
SIGABRT6Core进程调用abort()触发
SIGKILL9Term强制终止进程(无法捕获 / 阻塞 / 忽略
SIGSEGV11Core段错误(非法内存访问)
SIGPIPE13Term向无读者的管道写数据
SIGALRM14Term定时器信号(alarm()触发)
SIGTERM15Term终止信号(默认kill命令发送)
SIGSTOP19Stop暂停进程(无法捕获 / 阻塞 / 忽略

3. 信号相关函数

(1)发送信号:kill

向指定进程发送信号:

c

运行

#include <signal.h> #include <sys/types.h> // 向pid对应的进程发送sig信号 int kill(pid_t pid, int sig);
  • 参数:
    • pid:目标进程 PID;
    • sig:要发送的信号编号(如SIGKILL对应 9);
  • 返回值:成功返回 0,失败返回 - 1。
(2)捕获 / 自定义信号处理:signal

注册信号处理函数,自定义信号的行为:

c

运行

#include <signal.h> // 注册信号处理函数 void (*signal(int signum, void (*handler)(int)))(int);
  • 参数:
    • signum:要捕获的信号编号;
    • handler:处理函数(可选值:
      • SIG_DFL:默认处理;
      • SIG_IGN:忽略信号;
      • 自定义函数:如void my_handler(int sig));
  • 返回值:成功返回原处理函数指针,失败返回SIG_ERR

二、共享内存

共享内存是 System V 提供的一种高效的进程间通信方式,通过让多个进程直接访问同一块物理内存实现数据共享(无需拷贝,速度快)。

1. 共享内存的使用步骤

共享内存的生命周期分为 5 步:

  1. 申请共享内存
  2. 映射共享内存(将内核中的共享内存映射到进程地址空间);
  3. 读写共享内存
  4. 撤销映射
  5. 删除共享内存

2. 共享内存与管道的区别

特性共享内存管道
读写权限双方均可读写半双工(一端读、一端写)
同步机制需配合信号 / 信号量等同步自带同步(无数据时读阻塞)
数据管理不删除数据,数据保持数据读取后被移除
效率极高(直接访问内存)较低(需内核拷贝)

3. 共享内存相关函数

(1)创建 / 获取共享内存:shmget

c

运行

#include <sys/ipc.h> #include <sys/shm.h> // 创建或获取共享内存 int shmget(key_t key, size_t size, int shmflg);
  • 参数:
    • key:共享内存的键值(可通过ftok生成);
    • size:共享内存大小(字节);
    • shmflg:标志位(如IPC_CREAT|0666表示创建并设置权限);
  • 返回值:成功返回共享内存 ID,失败返回 - 1。
(2)生成键值:ftok

为共享内存生成唯一键值:

c

运行

#include <sys/ipc.h> key_t ftok(const char *pathname, int proj_id);
  • 参数:
    • pathname:已存在的文件路径;
    • proj_id:整形数字(通常用 ASCII 字符);
  • 返回值:成功返回键值,失败返回 - 1。
(3)映射共享内存:shmat

将共享内存映射到进程地址空间:

c

运行

#include <sys/shm.h> void *shmat(int shmid, const void *shmaddr, int shmflg);
  • 参数:
    • shmid:共享内存 ID;
    • shmaddr:映射地址(NULL 表示由系统分配);
    • shmflg:标志位(如 0 表示可读可写);
  • 返回值:成功返回映射后的地址,失败返回(void*)-1
(4)撤销映射:shmdt

将共享内存从进程地址空间撤销:

c

运行

#include <sys/shm.h> int shmdt(const void *shmaddr);
  • 参数:shmaddrshmat返回的映射地址;
  • 返回值:成功返回 0,失败返回 - 1。
(5)控制共享内存:shmctl

删除共享内存或获取 / 设置其属性:

c

运行

#include <sys/shm.h> int shmctl(int shmid, int cmd, struct shmid_ds *buf);
  • 参数:
    • shmid:共享内存 ID;
    • cmd:操作命令(如IPC_RMID表示删除共享内存);
    • buf:共享内存属性结构体(可传 NULL);
  • 返回值:成功返回 0,失败返回 - 1。

4. 共享内存相关命令

  • 查看共享内存:ipcs -m
  • 删除共享内存:ipcrm -m 共享内存ID

总结

  • 信号:适合异步通知、简单事件处理,缺点是无法传递复杂数据;
  • 共享内存:适合大量数据的高效共享,需配合同步机制(如信号、信号量)使用。
http://www.jsqmd.com/news/122541/

相关文章:

  • Electron 实战项目
  • LangFlow新闻摘要自动生成系统实战
  • 2025年年终北京别墅装修公司推荐:聚焦隐蔽工程与环保标准严苛评测,5家高口碑别墅专家深度解析 - 十大品牌推荐
  • Open-AutoGLM全局异常监听配置全攻略(避免线上事故的最后防线)
  • 2025 自动化运维产品深度对比:破解异构架构与合规高效双重挑战
  • 告别弹窗混乱时代,Open-AutoGLM实现自动化精准控制的4大秘诀
  • Open-AutoGLM异常处理全解析:5个必须掌握的解密容错策略
  • 2025年年终北京别墅装修公司推荐:基于多品牌深度横评与真实用户口碑的5家高可靠性服务商盘点 - 十大品牌推荐
  • LangFlow与主流大模型平台对比评测
  • Open-AutoGLM弹窗监控与自动关闭技术实践(独家方案首次公开)
  • 南京效果好质量优价格低全自动自清洗过滤器华博/博璟源优势解析 - 速递信息
  • 香港体育杰出领袖黑志宏领衔 沪港共筑海派武脉——绞龙棍激活香江青少年武术新生态
  • vxe-table 导入 excel xlsx 时,单元格内容值丢失前面0解决方法
  • 【紧急预警】Open-AutoGLM新版更新将加剧弹窗阻断,你做好准备了吗?
  • nanoid 全面教程:常用 API 串联与 Next.js 15 实战指南
  • 云迁移安全决胜之道:企业全业务上云的战略布局与风险防控指南
  • LangFlow异步执行支持情况说明
  • 1688商品详细信息价格SKU接口
  • nginx排查错误日志
  • 嵌入式第三十八篇——linux系统编程——IPC进程间通信
  • Vue3新特性实战:从入门到精通
  • springboot + vue
  • NAS网络存储:办公与数据管理的理想之选 - myqiye
  • 嵌入式第三十九篇——linux系统编程——信号通信、共享内存
  • Electron 文件系统与数据存储
  • 论文网站
  • LangFlow实战案例分享:构建多步骤推理AI代理全流程
  • 商用净水器大揭秘:流量大、服务好的品牌推荐 - 工业推荐榜
  • LangFlow多语言支持能力测试报告
  • 基于单片机的恒温储物箱设计(蓝牙RX TX反了)(有完整资料)