当前位置: 首页 > 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.cnnetsun.cn/news/178629.html

相关文章:

  • 【数据安全突围战】:Open-AutoGLM为何成为2024年最值得掌握的加密技术?
  • 使用机器学习简化机构沟通,提升可读性与包容性
  • LangFlow降低AI开发门槛:非技术人员也能构建智能应用
  • LangFlow与LangChain协同工作原理深度剖析
  • 16.2 对齐方法论:FineTune与RAG两大技术路径
  • 16.3 微调技术盘点:产品经理需要了解的核心方法
  • 汇编语言全接触-41.虚拟设备驱动程序初步
  • LangFlow能否实现专利文献摘要提取?科研情报处理
  • 告别熬夜爆肝:百考通AI如何用源码宝库与智能答辩重塑学习体验
  • AI赋能科研:百考通如何让学术起步更高效
  • LangFlow开源生态现状及未来发展方向预测
  • Open-AutoGLM自动化卡顿元凶分析(弹窗阻断深度解析与绕行策略)
  • 揭秘Open-AutoGLM运行时崩溃:为何弹窗错误始终无法捕获?
  • 【Open-AutoGLM加密传输协议配置】:掌握企业级安全通信的5大核心步骤
  • 2026毕设ssm+vue基于企业客户管理系统论文+程序
  • 【紧急故障应对】:Open-AutoGLM上线即超时?立即执行这6项止损操作
  • HoRain云--Java网络编程:BIO、NIO、AIO全解析
  • 基于java+ vue农产投入线上管理系统(源码+数据库+文档)
  • LangFlow能否用于构建智能客服质检系统?会话分析实践
  • Open-AutoGLM加载慢怎么办?3种高效优化策略立即见效
  • LangFlow能否用于构建AI心理咨询师原型?伦理边界探讨
  • Open-AutoGLM权限管理深度优化(90%开发者忽略的弹窗隐患)
  • 权限弹窗频发怎么办,一文掌握Open-AutoGLM无感授权处理方案
  • 12、便携式数字音频播放器选购指南
  • 20、数字音乐制作与优化全攻略
  • 9、Windows Vista数据安全与网络连接全攻略
  • 2、僵尸网络:行动的召唤
  • LangFlow中的冷启动问题缓解策略:默认模板推荐
  • LangFlow中的数据脱敏处理节点:隐私保护必备功能
  • 揭秘Open-AutoGLM跳转失败真相:3种高频场景的根治方案