学习笔记——Linux进程间通信(IPC)

Linux进程间通信(IPC)

一、信号通信

1. 信号概述

  • 应用场景

    • 异步通信

    • 通知机制

    • 处理随机事件

  • 特点:轻量级,用于进程间简单通知

2. 信号处理流程

  1. 信号产生:根据需要(随机事件)产生信号

  2. 内核处理:Linux内核接收到信号请求,在PCB链表中查找对应的PID

  3. 信号处理:找到对应进程后,暂停当前工作流程,执行PCB中信号处理函数

    • 例:发送信号2,则调用handle2()
  4. 恢复执行:信号处理函数执行完毕后,进程继续原来的代码执行

3. 信号相关函数

发送信号
复制代码
int kill(pid_t pid, int sig);
  • 功能:给指定进程发送信号

  • 参数

    • pid:接收信号的进程PID

    • sig:信号编号(可用kill -l查看)

  • 返回值:成功返回0,失败返回-1

信号捕获与处理
复制代码
void (*signal(int signum, void (*handler)(int)))(int);
// 或使用简化类型
sighandler_t signal(int signum, sighandler_t handler);
  • handler参数选项

    • SIG_DFL:默认处理

    • SIG_IGN:忽略信号

    • 自定义函数:用户自定义处理函数

查看信号信息
复制代码
man 7 signal  # 查看系统中信号的说明和默认处理行为

二、管道通信

1. 无名管道

复制代码
int pipe(int pipefd[2]);
  • 功能:创建并打开一个无名管道

  • 参数

    • pipefd[0]:固定读端

    • pipefd[1]:固定写端

  • 特点

    • 只能用于有亲缘关系的进程间通信

    • 单向通信

    • 生命周期随进程结束

2. 有名管道

复制代码
int mkfifo(const char *pathname, mode_t mode);
  • 功能:创建有名管道文件

  • 参数

    • pathname:管道文件路径+名称

    • mode:八进制文件权限

  • 特点

    • 可用于任意进程间通信

    • 以文件形式存在于文件系统

    • 需要手动删除

三、共享内存

1. 概述

  • 提供者:System V(Unix操作系统)

  • 特点

    • 最高效的IPC方式

    • 进程直接读写内存,无需内核介入

    • 需要配合其他同步机制(如信号、信号量集)

2. 与管道对比

特性 共享内存 管道
读写方向 双方都可读写 单向
读阻塞
写阻塞
数据存储 内存数组 内核缓冲区
数据保持 不删除数据 读取后删除

3. 使用步骤

复制代码
生成key → 申请对象 → 映射对象 → 读写对象 → 撤销映射 → 删除对象

4. 相关函数

生成唯一键值
复制代码
key_t ftok(const char *pathname, int proj_id);
  • 功能:生成唯一临时键值

  • 参数

    • pathname:任意存在的文件路径

    • proj_id:整形数字(通常用ASCII字符)

  • 注意:路径文件不能被删除重建

申请共享内存
复制代码
int shmget(key_t key, size_t size, int shmflg);
  • 功能:向内核申请共享内存

  • 参数

    • key:唯一键值

    • size:共享内存大小

    • shmflg:访问权限(八进制)+ 标志

      • IPC_CREAT:第一个申请时使用

      • IPC_EXCL:检测是否存在

  • 返回值:成功返回共享内存ID(shmid)

映射共享内存
复制代码
void *shmat(int shmid, const void *shmaddr, int shmflg);
  • 功能:将共享内存映射到本地内存空间

  • 参数

    • shmid:共享内存ID

    • shmaddr:本地地址(通常为NULL,系统自动分配)

    • shmflg

      • 0:可读写

      • SHM_RDONLY:只读

  • 返回值:成功返回映射地址

读写操作
复制代码
memcpy();  // 二进制数据
strcpy();  // 字符串数据
撤销映射
复制代码
int shmdt(const void *shmaddr);
  • 功能:断开本地内存与共享内存的映射

  • 参数shmaddr - 映射地址

删除共享内存对象
复制代码
int shmctl(int shmid, int cmd, struct shmid_ds *buf);
  • 功能:修改属性或删除共享内存

  • 参数

    • shmid:共享内存ID

    • cmdIPC_RMID(删除)

    • buf:NULL(只删除对象)

5. 系统命令

复制代码
ipcs -a      # 查询所有IPC对象(共享内存、信号量集、消息队列)
ipcrm -m ID  # 删除指定的共享内存

四、使用注意事项

  1. 信号

    • 信号处理函数应尽量简短

    • 注意信号的可重入性问题

    • 某些信号不可捕获(如SIGKILL)

  2. 管道

    • 无名管道需在fork前创建

    • 有名管道需要处理读写阻塞

    • 注意管道缓冲区大小限制

  3. 共享内存

    • 必须配合同步机制使用

    • 注意内存映射的生命周期

    • 及时清理,避免内存泄漏

    • 注意多进程并发访问的数据一致性问题

  4. 通用建议

    • 错误处理要完善

    • 资源使用后及时释放

    • 考虑进程异常退出的清理工作

相关推荐
Sammyyyyy1 天前
Rust 1.92.0 发布:Never Type 进一步稳定
java·算法·rust
报错小能手1 天前
线程池学习(七)实现定时(调度)线程池
学习
alonewolf_991 天前
深入解析G1与ZGC垃圾收集器:原理、调优与选型指南
java·jvm·算法
weixin_399380691 天前
OA 系统假死问题分析与优化
java·运维
SmartRadio1 天前
在CH585M代码中如何精细化配置PMU(电源管理单元)和RAM保留
linux·c语言·开发语言·人工智能·单片机·嵌入式硬件·lora
济6171 天前
linux(第十四期)--Uboot移植(2)-- 在U-Boot 中添加自己的开发板-- Ubuntu20.04
linux·运维·服务器
●VON1 天前
从模型到价值:MLOps 工程体系全景解析
人工智能·学习·制造·von
好奇龙猫1 天前
【人工智能学习-AI-MIT公开课第 18. 表示:分類、軌跡、過渡】
学习
ben9518chen1 天前
嵌入式linux操作系统简介
linux·运维·服务器
菜鸟笔记本1 天前
linux设置定时备份mysql数据
linux·mysql·oracle