c++实现在同一台主机两个程序实现实时通信

server/2025/3/9 18:29:34/

c++代码实现同一台主机两个程序实现实时通信

对于进程间通信方式有很多种,对于本机两个程序中可以实时通信那么应该选择哪个呢?很多人最快想到的就是socket通信。

套接字适用于跨网络的进程间通信,具有网络透明性、灵活性和安全性高的优点,但性能开销较大,延迟较高!

共享内存适用于同一台机器上的进程间通信,具有高性能、低延迟和高吞吐量的优点,但需要手动管理同步机制,安全性较低。所以我们优先使用共享内存来实现,在智能驾驶等工业场景中使用非常多。

共享内存实现

进程A:

#include <cstring>
#include <iostream>
#include <sys/shm.h>
#include <thread>
#include <unistd.h>#define SHM_KEY_A 12345
#define SHM_KEY_B 67890
#define SHM_SIZE 1024void send_messages(char* send_data)
{std::string message;while (std::getline(std::cin, message)) {if (message == "exit") {break;}strncpy(send_data, message.c_str(), SHM_SIZE - 1);send_data[SHM_SIZE - 1] = '\0';while (std::strlen(send_data) > 0) {usleep(100000); // 等待 100 毫秒}}
}void receive_messages(char* recv_data)
{while (true) {while (std::strlen(recv_data) == 0) {usleep(100000); // 等待 100 毫秒}std::cout << "Received message: " << recv_data << std::endl;std::memset(recv_data, 0, SHM_SIZE);}
}int main()
{// 创建共享内存A(用于接收消息)int shmid_a = shmget(SHM_KEY_A, SHM_SIZE, 0666);if (shmid_a == -1) {std::cerr << "Failed to get shared memory A" << std::endl;return 1;}// 创建共享内存B(用于发送消息)int shmid_b = shmget(SHM_KEY_B, SHM_SIZE, 0666);if (shmid_b == -1) {std::cerr << "Failed to get shared memory B" << std::endl;return 1;}// 将共享内存连接到当前进程char* recv_data = (char*)shmat(shmid_a, nullptr, 0);char* send_data = (char*)shmat(shmid_b, nullptr, 0);if (recv_data == (char*)-1 || send_data == (char*)-1) {std::cerr << "Failed to attach shared memory" << std::endl;return 1;}std::thread sender(send_messages, send_data);std::thread receiver(receive_messages, recv_data);sender.join();receiver.join();// 清理共享内存shmdt(recv_data);shmdt(send_data);shmctl(shmid_a, IPC_RMID, nullptr);shmctl(shmid_b, IPC_RMID, nullptr);return 0;
}

进程B:

#include <cstring>
#include <iostream>
#include <sys/shm.h>
#include <thread>
#include <unistd.h>#define SHM_KEY_A 12345
#define SHM_KEY_B 67890
#define SHM_SIZE 1024void send_messages(char* send_data)
{std::string message;while (std::getline(std::cin, message)) {if (message == "exit") {break;}strncpy(send_data, message.c_str(), SHM_SIZE - 1);send_data[SHM_SIZE - 1] = '\0';while (std::strlen(send_data) > 0) {usleep(100000); // 等待 100 毫秒}}
}void receive_messages(char* recv_data)
{while (true) {while (std::strlen(recv_data) == 0) {usleep(100000); // 等待 100 毫秒}std::cout << "Received message: " << recv_data << std::endl;std::memset(recv_data, 0, SHM_SIZE);}
}int main()
{// 获取共享内存A(用于发送消息)int shmid_a = shmget(SHM_KEY_A, SHM_SIZE, IPC_CREAT | 0666);if (shmid_a == -1) {std::cerr << "Failed to create shared memory A" << std::endl;return 1;}// 获取共享内存B(用于接收消息)int shmid_b = shmget(SHM_KEY_B, SHM_SIZE, IPC_CREAT | 0666);if (shmid_b == -1) {std::cerr << "Failed to create shared memory B" << std::endl;return 1;}// 将共享内存连接到当前进程char* send_data = (char*)shmat(shmid_a, nullptr, 0);char* recv_data = (char*)shmat(shmid_b, nullptr, 0);if (send_data == (char*)-1 || recv_data == (char*)-1) {std::cerr << "Failed to attach shared memory" << std::endl;return 1;}std::thread sender(send_messages, send_data);std::thread receiver(receive_messages, recv_data);sender.join();receiver.join();// 清理共享内存shmdt(send_data);shmdt(recv_data);shmctl(shmid_a, IPC_RMID, nullptr);shmctl(shmid_b, IPC_RMID, nullptr);return 0;
}

代码简洁比较容易。有些捞仔可能会问了,除了共享内存还有没有其它方式呢?有!当然有,只不过性能不如共享内存啦。

命名管道实现

进程A:

#include <fstream>
#include <iostream>
#include <sys/stat.h>
#include <thread>
#include <unistd.h>#define PIPE_NAME_A "pipe_a"
#define PIPE_NAME_B "pipe_b"void send_messages(const std::string& pipe_name)
{std::ofstream pipe(pipe_name);if (!pipe) {std::cerr << "Failed to open named pipe for writing" << std::endl;return;}std::cout << "Enter messages to send (type 'exit' to quit):" << std::endl;while (true) {std::string message;std::getline(std::cin, message);if (message == "exit") {break;}pipe << message << std::endl;pipe.flush();}
}void receive_messages(const std::string& pipe_name)
{std::ifstream pipe(pipe_name);if (!pipe) {std::cerr << "Failed to open named pipe for reading" << std::endl;return;}std::cout << "Waiting for messages..." << std::endl;while (true) {std::string message;std::getline(pipe, message);// 处理程序在没有数据时不会频繁地检查管道,从而节省CPU资源,同时保持对新数据的响应if (pipe.eof()) {sleep(1);pipe.clear();pipe.seekg(0, std::ios::end);} else {std::cout << "Received message: " << message << std::endl;}}
}int main()
{// 创建命名管道if (mkfifo(PIPE_NAME_A, 0666) == -1) {std::cerr << "Failed to create named pipe A" << std::endl;return 1;}if (mkfifo(PIPE_NAME_B, 0666) == -1) {std::cerr << "Failed to create named pipe B" << std::endl;return 1;}std::thread sender(send_messages, PIPE_NAME_A);std::thread receiver(receive_messages, PIPE_NAME_B);sender.join();receiver.join();return 0;
}

进程B:

#include <fstream>
#include <iostream>
#include <sys/stat.h>
#include <thread>
#include <unistd.h>#define PIPE_NAME_A "pipe_a"
#define PIPE_NAME_B "pipe_b"void send_messages(const std::string& pipe_name)
{std::ofstream pipe(pipe_name);if (!pipe) {std::cerr << "Failed to open named pipe for writing" << std::endl;return;}std::cout << "Enter messages to send (type 'exit' to quit):" << std::endl;while (true) {std::string message;std::getline(std::cin, message);if (message == "exit") {break;}pipe << message << std::endl;pipe.flush();}
}void receive_messages(const std::string& pipe_name)
{std::ifstream pipe(pipe_name);if (!pipe) {std::cerr << "Failed to open named pipe for reading" << std::endl;return;}std::cout << "Waiting for messages..." << std::endl;while (true) {std::string message;std::getline(pipe, message);if (pipe.eof()) {sleep(1);pipe.clear();pipe.seekg(0, std::ios::end);} else {std::cout << "Received message: " << message << std::endl;}}
}int main()
{std::thread sender(send_messages, PIPE_NAME_B);std::thread receiver(receive_messages, PIPE_NAME_A);sender.join();receiver.join();return 0;
}

相对于共享内存的优缺点:

优点

  • 简单易用:命名管道的实现相对简单,适合用于简单的进程间通信。

缺点

  1. 性能开销:命名管道的性能略低于共享内存,因为数据需要通过内核缓冲区传输。共享内存是所有进程间通信(IPC)机制中最快的,因为它不需要在进程之间复制数据。
  2. 单向通信:命名管道通常是单向的,需要两个管道实现双向通信。

还可以使用消息队列实现(本人使用的是wsl2,微软不支持,故没有代码)


http://www.ppmy.cn/server/173731.html

相关文章

MySQL如何给其他账号分配权限?

目录 基础权限分配流程 权限级别对照表 安全事项 MySQL数据库使用root账号为test账号分配权限的标准操作流程及注意事项&#xff1a; 基础权限分配 ‌1、root用户登录, 输入root密码后进入MySQL命令行环境‌ mysql -u root -p2、用户存在性校验 SELECT user,host FROM …

深度生成模型(五)——生成对抗网络 GAN

与传统自编码器&#xff08;AE&#xff09;相比&#xff0c;变分自编码器&#xff08;VAE&#xff09;在图像生成任务中能够实现更高的图像质量和更丰富的多样性。这主要归功于其利用隐空间中的均值和方差参数生成样本&#xff0c;并通过KL散度在训练过程中对隐空间进行正则化&…

蓝桥杯历年真题题解

1.轨道炮&#xff08;数学模拟&#xff09; #include <iostream> #include <map> using namespace std; const int N1010; int x[N],y[N],v[N]; char d[N]; int main() {int n;int ans-100;cin>>n;for(int i1;i<n;i)cin>>x[i]>>y[i]>>v…

面试时,如何回答好“你是怎么测试接口的?”

一、回答框架设计&#xff08;金字塔原理&#xff09; 总述&#xff1a; "我通常采用分层测试策略&#xff0c;遵循需求分析→场景拆解→用例设计→自动化实施→持续监控的闭环流程。以最近测试的支付中台项目为例&#xff0c;核心围绕功能正确性、异常鲁棒性、安全防护、…

源码:用Python进行电影数据分析实战指南

源码&#xff1a;用Python进行电影数据分析实战指南 原创 IT小本本 IT小本本 2025年03月03日 22:28 北京 接上一篇文章&#xff1a;用Python进行电影数据分析实战指南 1、首先复制csv内容到csv文件中 2、接着创建.py文件复制源码内容 3、运行代码&#xff0c;就可以看到数据…

如何搭建个人静态住宅IP:从零开始

你好&#xff01;今天我们将一起探索如何从头开始搭建个人静态住宅IP。无论您是为了远程办公、在线教育还是游戏加速&#xff0c;静态住宅IP都能带给您更稳定的网络体验。 一、准备阶段 1. 明确需求 首先&#xff0c;您需要清楚自己为什么需要静态住宅IP。可能是为了实现远程…

ZT39 小红的取模构造

描述 小红拿到了两个整数 a 和 b 。她希望你构造两个正整数 x 和 y 满足&#xff1a; 1、x mod ya 2、y mod xb 其中 p mod q 代表 p 除以 q 得到的余数。 共有t组询问。 输入描述&#xff1a; 第一行输入一个正整数t&#xff0c;代表询问的次数。 接下来的t行&#xff0c;每行…

IPD(集成产品开发)简介

参考&#xff1a;IPD咨询_研发管理咨询_IPD集成产品开发-百思特管理咨询集团 一、什么是IPD IPD到底是什么&#xff1f;一套体系&#xff1f;一些流程&#xff1f;还是一种模式&#xff1f; 华为在整个企业内部改革中最重要的两个项目一个是ISC(集成供应链)&#xff0c;另外…