C++在Linux下如何进行进程间通信
C++在Linux下如何进行进程间通信

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈
在Linux环境下开发C++程序,进程间通信(IPC)是个绕不开的话题。当多个进程需要协同工作、交换数据时,Linux系统提供了多种成熟的方案。每种方案都有其独特的“性格”和适用场景,理解它们的差异,是写出高效、稳定程序的关键。接下来,我们就逐一拆解这些常用的IPC方法。
1. 管道(Pipes)
管道算得上是Linux IPC的“元老”了,它提供了一种半双工的通信方式。什么叫半双工?简单说,就是数据只能单向流动。这种机制在父子进程之间尤其好用,因为它直接继承了文件描述符,创建和使用都非常直观。
无名管道(Anonymous Pipes)
无名管道,顾名思义,它没有名字,生命周期随进程结束而终结。它最大的特点就是只能在具有亲缘关系的进程(比如父子进程)间使用。其工作原理是:先调用pipe()系统调用创建一个管道,这会返回两个文件描述符,一个用于读,一个用于写。父进程fork()出子进程后,双方各自关闭不需要的一端,就能建立起一条单向的数据通道。
#include
#include
#include
#include
int main() {
int pipefd[2];
pid_t pid;
char buffer[256];
// 创建管道
if (pipe(pipefd) == -1) {
perror("pipe");
exit(EXIT_FAILURE);
}
pid = fork();
if (pid == -1) {
perror("fork");
exit(EXIT_FAILURE);
}
if (pid == 0) { // 子进程
close(pipefd[1]); // 关闭写端
read(pipefd[0], buffer, sizeof(buffer));
std::cout << "Child received: " << buffer << std::endl;
close(pipefd[0]);
} else { // 父进程
close(pipefd[0]); // 关闭读端
const char* message = "Hello from parent";
write(pipefd[1], message, strlen(message) + 1);
close(pipefd[1]);
wait(NULL); // 等待子进程结束
}
return 0;
}
命名管道(Named Pipes, FIFOs)
无名管道好用,但限制也明显——只能用于亲缘进程。这时候,命名管道(FIFO)就派上用场了。它在文件系统中有一个路径名,任何知道这个路径的进程都可以打开它进行读写,完全解除了亲缘关系的束缚。你可以把它想象成一个有名字的、先进先出的特殊文件。
#include
#include
#include
#include
#include
int main() {
const char* fifo_name = "/tmp/myfifo";
int fd;
char buffer[256];
// 创建命名管道
if (mkfifo(fifo_name, 0666) == -1) {
perror("mkfifo");
exit(EXIT_FAILURE);
}
fd = open(fifo_name, O_RDWR);
if (fd == -1) {
perror("open");
exit(EXIT_FAILURE);
}
// 写入数据
const char* message = "Hello from FIFO";
write(fd, message, strlen(message) + 1);
// 读取数据
read(fd, buffer, sizeof(buffer));
std::cout << "Received: " << buffer << std::endl;
close(fd);
unlink(fifo_name); // 删除命名管道
return 0;
}
2. 消息队列(Message Queues)
如果说管道是字节流,那么消息队列就是“快递柜”。它允许进程以结构化的消息为单位进行数据交换,每条消息都有类型标识。发送方和接收方可以基于消息类型进行选择性接收,这提供了比管道更灵活的通信模式。消息队列独立于进程存在,即使通信进程结束了,消息仍然可以保留在队列中。
#include
#include
#include
#include
#include
struct msg_buffer {
long mtype;
char mtext[100];
};
int main() {
key_t key = ftok("msgqueue_example.c", 65);
int msgid = msgget(key, 0666 | IPC_CREAT);
if (msgid == -1) {
perror("msgget");
exit(EXIT_FAILURE);
}
msg_buffer buffer;
buffer.mtype = 1;
strcpy(buffer.mtext, "Hello from message queue");
// 发送消息
if (msgsnd(msgid, &buffer, sizeof(buffer.mtext), 0) == -1) {
perror("msgsnd");
exit(EXIT_FAILURE);
}
// 接收消息
if (msgrcv(msgid, &buffer, sizeof(buffer.mtext), 1, 0) == -1) {
perror("msgrcv");
exit(EXIT_FAILURE);
}
std::cout << "Received message: " << buffer.mtext << std::endl;
msgctl(msgid, IPC_RMID, NULL); // 删除消息队列
return 0;
}
3. 共享内存(Shared Memory)
追求极致性能?那共享内存绝对是你的首选。它的原理很直接:在内存中划出一块区域,让多个进程都能映射到自己的地址空间。这样一来,进程间的数据交换就变成了直接对内存的读写,完全避免了内核缓冲区的数据拷贝,速度自然是最快的。但天下没有免费的午餐,共享内存不提供任何同步机制,需要配合信号量等工具来防止数据竞争。
#include
#include
#include
#include
#include
int main() {
key_t key = ftok("shm_example.c", 65);
int shmid = shmget(key, 1024, 0666 | IPC_CREAT);
if (shmid == -1) {
perror("shmget");
exit(EXIT_FAILURE);
}
char* str = (char*) shmat(shmid, (void*)0, 0);
if (str == (char*)(-1)) {
perror("shmat");
exit(EXIT_FAILURE);
}
strcpy(str, "Hello from shared memory");
std::cout << "Message written in memory: " << str << std::endl;
shmdt(str); // 分离共享内存
// 删除共享内存
shmctl(shmid, IPC_RMID, NULL);
return 0;
}
4. 信号量(Semaphores)
严格来说,信号量本身并不是为了传递数据,而是为了解决一个更根本的问题:进程同步。当多个进程需要访问同一块共享资源(比如刚提到的共享内存)时,如何避免冲突?信号量就像一个“资源计数器”或“通行证”。最常见的二元信号量(互斥锁)可以确保同一时刻只有一个进程能进入临界区。P操作(等待)尝试获取信号量,V操作(发信号)则释放信号量。
#include
#include
#include
#include
#include
union semun {
int val;
struct semid_ds *buf;
unsigned short *array;
};
int main() {
key_t key = ftok("semaphore_example.c", 65);
int semid = semget(key, 1, 0666 | IPC_CREAT);
if (semid == -1) {
perror("semget");
exit(EXIT_FAILURE);
}
union semun arg;
arg.val = 1; // 初始化信号量为1
if (semctl(semid, 0, SETVAL, arg) == -1) {
perror("semctl");
exit(EXIT_FAILURE);
}
// P操作(等待信号量)
struct sembuf sb = {0, -1, SEM_UNDO};
if (semop(semid, &sb, 1) == -1) {
perror("semop");
exit(EXIT_FAILURE);
}
std::cout << "Semaphore P operation completed" << std::endl;
// V操作(释放信号量)
sb.sem_op = 1;
if (semop(semid, &sb, 1) == -1) {
perror("semop");
exit(EXIT_FAILURE);
}
std::cout << "Semaphore V operation completed" << std::endl;
semctl(semid, 0, IPC_RMID); // 删除信号量
return 0;
}
5. 套接字(Sockets)
最后登场的是功能最强大的套接字。它最初是为网络通信设计的,但同样完美适用于同一台主机上的进程间通信(通过Unix Domain Sockets)。套接字提供全双工通信,支持多种协议(如TCP的可靠流、UDP的数据报),并且是跨平台兼容性最好的IPC方式之一。如果未来你的程序有扩展到网络通信的可能,那么从一开始就使用套接字会是一个具有前瞻性的选择。
Unix Domain Sockets
这是专门用于本地IPC的套接字,它通过文件系统路径名来标识,效率比网络套接字更高,因为它避免了网络协议栈的开销。
#include
#include
#include
#include
#include
#include
int main() {
struct sockaddr_un addr;
int sockfd, connfd;
char buffer[1024];
sockfd = socket(AF_UNIX, SOCK_STREAM, 0);
if (sockfd == -1) {
perror("socket");
exit(EXIT_FAILURE);
}
memset(&addr, 0, sizeof(addr));
addr.sun_family = AF_UNIX;
strncpy(addr.sun_path, "/tmp/unix_socket", sizeof(addr.sun_path) - 1);
unlink("/tmp/unix_socket"); // 删除已存在的套接字文件
if (bind(sockfd, (struct sockaddr*)&addr, sizeof(addr)) == -1) {
perror("bind");
exit(EXIT_FAILURE);
}
if (listen(sockfd, 5) == -1) {
perror("listen");
exit(EXIT_FAILURE);
}
connfd = accept(sockfd, NULL, NULL);
if (connfd == -1) {
perror("accept");
exit(EXIT_FAILURE);
}
read(connfd, buffer, sizeof(buffer));
std::cout << "Received: " << buffer << std::endl;
close(connfd);
close(sockfd);
return 0;
}
好了,以上就是Linux下C++常用的几种进程间通信方法。简单总结一下:管道简单直接,适合亲缘进程;命名管道打破了亲缘限制;消息队列提供了结构化的消息传递;共享内存速度最快,但需手动同步;信号量是同步的基石;而套接字则是最通用、最强大的方案。实际开发中,没有绝对的“最佳”,只有最“合适”。理解它们的原理和特性,根据你的具体场景——是要求速度、简便性,还是未来的可扩展性——来做出选择,这才是关键所在。
游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。
同类文章
Sublime写前端代码卡顿怎么办?优化Sublime运行速度的几个设置
Sublime写前端代码卡顿怎么办?优化Sublime运行速度的几个设置 有没有遇到过这种情况?明明只是敲几行代码,Sublime Text却突然变得一卡一卡的,光标移动都带着延迟。别急着怀疑电脑性能,问题很可能出在编辑器本身的设置上。尤其是前端项目,动辄成千上万的依赖文件,对编辑器来说是个不小的负
VSCode怎么把选中的大段代码片段一键提取为一个新的函数或方法
VSCode Extract Function:选中代码前必须确认三件事 想把一段代码变成独立的函数?VSCode的Extract Function功能确实能一键搞定。但先别急着操作,这个功能可不会猜你的心思,它只认“语法上合法的选中区域”。很多时候操作失败,并不是功能坏了,而是你选中的代码块本身就
VSCode远程开发断连_解决SSH连接超时与自动重连
根本原因是服务器端sshd空闲超时或防火墙 NAT丢弃连接,需两端配合:服务端启用ClientAliveInterval 60和ClientAliveCountMax 3并重启sshd,客户端检查~ ssh config避免冲突,同时清理VSCode残留进程并更新扩展。 SSH连接被服务器主动断开
VSCode项目资源管理器排序_按类型或修改时间排列文件
VSCode资源管理器默认按什么排序? 当你打开VSCode的资源管理器,看着里面一堆文件,有没有好奇过它们的排列顺序?默认情况下,VSCode采用的是最“老实”的规则:按文件名的字母顺序排列,并且不区分大小写。这意味着,index js之所以排在README md前面,仅仅是因为字母“i”在字母表
Sublime怎么配置Kotlin环境?Sublime编写Kotlin代码高亮设置
Sublime Text 不能直接运行 Kotlin,但可通过安装 Kotlin 插件实现语法高亮,并配置构建系统调用 kotlinc 和 ja va 编译运行;前提是 JDK 与 Kotlin 编译器已正确安装并加入 PATH,且构建系统中 selector 必须为 source kotlin、-
- 日榜
- 周榜
- 月榜
1
2
3
4
5
6
7
8
9
10
相关攻略
2015-03-10 11:25
2015-03-10 11:05
2021-08-04 13:30
2015-03-10 11:22
2015-03-10 12:39
2022-05-16 18:57
2025-05-23 13:43
2025-05-23 14:01
热门教程
- 游戏攻略
- 安卓教程
- 苹果教程
- 电脑教程
热门话题

