一、概述
在 Linux 系统中,进程间通信(IPC)是实现多进程协作的核心能力,其中信号通信 和共享内存是两种高频使用的通信方式:
- 信号通信:主打 "异步通知",适用于进程间的事件触发、状态唤醒等场景;
- 共享内存:是最快的 IPC 方式,通过共享物理内存实现数据互通,需搭配信号 / 信号量实现同步。
二、信号通信:异步通知的核心机制
2.1 信号的核心定位
信号是 Linux 内核提供的异步通信机制,本质是 "通知机制",用于处理系统中的 "随机事件"(如进程暂停、唤醒、终止、自定义事件等),核心特点:
- 异步性:信号的产生和处理与进程主流程无固定时序;
- 中断性:信号到达时,进程会暂停当前流程,优先执行信号处理函数,执行完毕后恢复原流程。
2.2 信号发送与接收的完整流程
- 信号产生 :由随机事件触发(如
kill命令、系统调用、硬件异常等); - 内核查找目标进程:Linux 内核接收到信号发送请求后,在进程控制块(PCB)的信号链表中,查找目标 PID 对应的进程;
- 中断并执行处理函数 :找到目标进程后,暂停其原有工作流程,执行 PCB 中信号编号对应下标的处理函数(如信号 2 对应
handle2); - 恢复原流程:信号处理函数执行完毕后,进程回到原有代码继续运行。

2.3 信号相关核心函数
(1)发送信号:kill ()
c
运行
#include <signal.h>
int kill(pid_t pid, int sig);
- 功能:向指定 PID 的进程发送指定编号的信号;
- 参数 :
pid:接收信号的进程 PID;sig:信号编号(可通过kill -l查看所有信号编号);
- 返回值:成功返回 0,失败返回 - 1。
示例:向 PID 为 1000 的进程发送 SIGCONT(唤醒)信号
c
运行
kill(1000, 18); // 18是SIGCONT的默认编号,等价于kill -CONT 1000
(2)捕获并自定义信号处理:signal ()
c
运行
#include <signal.h>
// 函数原型(简化版)
sighandler_t signal(int signum, sighandler_t handler);
- 功能:注册信号处理函数,自定义信号的处理行为;
- 参数 :
signum:要捕获的信号编号;handler:信号处理方式,支持 3 种:SIG_DFL:使用系统默认处理行为;SIG_IGN:忽略该信号;- 自定义函数:如
void myhandle(int num),接收信号编号作为参数;
- 返回值 :成功返回原信号处理函数地址,失败返回
SIG_ERR。
示例:自定义 SIGCONT 信号的处理函数
c
运行
void myhandle(int num) {
printf("捕获到信号%d,进程被唤醒\n", num);
}
// 注册信号处理函数
signal(SIGCONT, myhandle);
2.4 信号相关辅助命令
- 查看所有信号的编号和名称:
kill -l; - 查看信号的详细说明和默认处理行为:
man 7 signal。
三、共享内存:最快的进程间通信方式
3.1 共享内存的核心定位
共享内存是 System V 标准提供的 IPC 方式,核心是让多个进程映射同一块物理内存到自己的地址空间,实现数据直接互通。
- 优势:无需数据拷贝,是所有 IPC 中速度最快的;
- 注意:共享内存本身无同步 / 互斥机制,需搭配信号、信号量等实现 "读写同步"。
3.2 共享内存的使用流程(核心 6 步)
graph LR
A[生成唯一Key值:ftok()] --> B[申请共享内存:shmget()]
B --> C[映射到进程地址空间:shmat()]
C --> D[读写共享内存:memcpy/strcpy]
D --> E[撤销映射:shmdt()]
E --> F[删除共享内存:shmctl()]
3.3 共享内存核心函数
(1)生成唯一 Key 值:ftok ()
c
运行
#include <sys/ipc.h>
key_t ftok(const char *pathname, int proj_id);
- 功能:通过文件路径和自定义标识生成唯一键值,用于标识共享内存;
- 参数 :
pathname:任意文件路径(需保证文件不被删除 / 重建,否则 Key 值会变化);proj_id:整型标识(通常用 ASCII 单字符,如'!');
- 返回值:成功返回唯一 Key 值,失败返回 - 1。
示例:
c
运行
key_t key = ftok("./", '!'); // 基于当前目录生成Key值
(2)申请共享内存:shmget ()
c
运行
#include <sys/shm.h>
int shmget(key_t key, size_t size, int shmflg);
- 功能:向内核申请指定大小的共享内存;
- 参数 :
key:ftok 生成的唯一 Key 值;size:共享内存大小(字节,建议为 4096 的整数倍);shmflg:权限 + 创建标识,常用IPC_CREAT | 0666(不存在则创建,权限为 666);
- 返回值:成功返回共享内存 ID(shmid),失败返回 - 1。
示例:
c
运行
int shmid = shmget(key, 4096, IPC_CREAT | 0666);
(3)映射共享内存:shmat ()
c
运行
#include <sys/shm.h>
void *shmat(int shmid, const void *shmaddr, int shmflg);
- 功能:将共享内存映射到进程的地址空间;
- 参数 :
shmid:shmget 返回的共享内存 ID;shmaddr:指定映射地址,NULL 表示由系统自动分配;shmflg:访问权限,0 表示可读写,SHM_RDONLY表示只读;
- 返回值 :成功返回映射地址,失败返回
(void*)-1。
示例:
c
运行
void *p = shmat(shmid, NULL, 0); // 映射为可读写
(4)读写共享内存
共享内存映射后可直接当作普通内存使用,支持字符串 / 二进制数据操作:
c
运行
// 写入字符串
strcpy((char*)p, "共享内存测试数据");
// 读取字符串
printf("共享内存内容:%s\n", (char*)p);
// 二进制数据读写(如结构体)
memcpy(p, &data, sizeof(data));
(5)撤销映射:shmdt ()
c
运行
#include <sys/shm.h>
int shmdt(const void *shmaddr);
- 功能:断开进程与共享内存的映射关系(仅解绑,不删除共享内存);
- 参数:shmat 返回的映射地址;
- 返回值:成功返回 0,失败返回 - 1。
示例:
c
运行
shmdt(p); // 撤销映射
(6)删除共享内存: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。
示例:
c
运行
shmctl(shmid, IPC_RMID, NULL); // 彻底删除共享内存
3.4 共享内存相关命令
- 查看系统中所有共享内存、信号量、消息队列:
ipcs -a; - 删除指定 ID 的共享内存:
ipcrm -m 共享内存ID。
四、共享内存与管道(无名 / 有名)的核心区别
管道(无名pipe/ 有名mkfifo)也是常用 IPC 方式,但与共享内存差异显著:
| 特性 | 共享内存 | 管道(无名 / 有名) |
|---|---|---|
| 读写权限 | 双方均可读写 | 无名管道:固定读端 / 写端;有名管道:双向但需同步 |
| 阻塞特性 | 无读 / 写阻塞 | 读阻塞(无数据)、写阻塞(缓冲区满) |
| 同步机制 | 无,需搭配信号 / 信号量 | 自带同步(阻塞机制) |
| 数据存储 | 内存中,不删除则一直存在 | 内核缓冲区,读取后数据消失 |
| 数据拷贝 | 无拷贝(直接操作内存) | 需内核态 / 用户态拷贝 |
| 易用性 | 需手动管理映射 / 删除 | 可当作文件操作,更简单 |
管道核心函数补充
(1)创建无名管道:pipe ()
c
运行
#include <unistd.h>
int pipe(int pipefd[2]);
- 功能:创建并打开无名管道;
- 参数 :
pipefd[0]为读端,pipefd[1]为写端; - 返回值:成功返回 0,失败返回 - 1。
(2)创建有名管道:mkfifo ()
c
运行
#include <sys/stat.h>
int mkfifo(const char *pathname, mode_t mode);
- 功能:创建有名管道文件;
- 参数 :
pathname:管道文件的路径 + 名称;mode:文件权限(8 进制,如 0666);
- 返回值:成功返回 0,失败返回 - 1。
五、完整示例:共享内存 + 信号实现进程通信
5.1 进程 A:创建共享内存,写入数据,等待信号唤醒
c
运行
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
void myhandle(int num) {
printf("进程A捕获到信号%d,被唤醒\n", num);
}
int main() {
// 1. 生成Key值
key_t key = ftok("./", '!');
if (key == -1) { perror("ftok"); return 1; }
// 2. 申请共享内存
int shmid = shmget(key, 4096, IPC_CREAT | 0666);
if (shmid == -1) { perror("shmget"); return 1; }
// 3. 映射共享内存
void *p = shmat(shmid, NULL, 0);
if (p == (void*)-1) { perror("shmat"); return 1; }
// 4. 写入数据
strcpy((char*)p, "Hello, 共享内存+信号通信");
printf("进程A PID:%d,已写入数据到共享内存\n", getpid());
// 5. 注册SIGCONT信号处理函数
signal(SIGCONT, myhandle);
// 6. 等待信号唤醒
printf("进程A进入阻塞,等待信号...\n");
pause();
// 7. 撤销映射
shmdt(p);
// 8. 删除共享内存(可选)
shmctl(shmid, IPC_RMID, NULL);
return 0;
}
5.2 进程 B:读取共享内存,发送信号唤醒进程 A
c
运行
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
int main(int argc, char *argv[]) {
if (argc < 2) {
printf("用法:%s <进程A的PID>\n", argv[0]);
return 1;
}
pid_t pid_a = atoi(argv[1]);
// 1. 生成相同的Key值
key_t key = ftok("./", '!');
if (key == -1) { perror("ftok"); return 1; }
// 2. 获取共享内存
int shmid = shmget(key, 4096, 0666);
if (shmid == -1) { perror("shmget"); return 1; }
// 3. 映射共享内存
void *p = shmat(shmid, NULL, 0);
if (p == (void*)-1) { perror("shmat"); return 1; }
// 4. 读取共享内存数据
printf("进程B读取到共享内存数据:%s\n", (char*)p);
// 5. 发送SIGCONT信号唤醒进程A
kill(pid_a, 18);
printf("进程B已向进程A发送唤醒信号\n");
// 6. 撤销映射
shmdt(p);
return 0;
}
5.3 运行步骤
- 编译进程 A:
gcc shm_signal_a.c -o a.out,运行:./a.out(记录进程 A 的 PID); - 编译进程 B:
gcc shm_signal_b.c -o b.out,运行:./b.out <进程A的PID>; - 观察进程 A 输出:捕获到信号 18,被唤醒,完成通信。
六、总结
- 信号通信 :核心是 "异步通知",通过
kill发送信号、signal捕获信号,适用于事件触发、进程唤醒等场景; - 共享内存:最快的 IPC 方式,核心流程是 "Key→申请→映射→读写→解绑→删除",需搭配信号 / 信号量实现同步;
- 与管道对比:共享内存无阻塞、无数据拷贝,但需手动管理;管道易用性高,自带同步但速度慢;
- 实际开发中,共享内存 + 信号是高性能进程通信的常用组合,既保证数据传输效率,又能实现事件同步。
