自动驾驶(八十八)---------通讯之SOMEIP

1. 什么是SOME/IP

        服务导向架构(SOA,Service-Oriented Architecture)是一种设计软件系统的方法,强调通过可重用的服务来实现系统的松散耦合。每个服务是独立的功能单元,可以被不同的应用程序使用。这些服务通过标准化的接口和协议进行通信和数据交换。其特点是:

  1. 松散耦合:服务之间是独立的,减少了组件之间的依赖性,便于系统的扩展和维护。
  2. 可重用性:服务可以被多个应用程序重复使用,提高了代码的重用性和开发效率。
  3. 互操作性:基于标准化的接口和协议(如SOAP、REST、SOME/IP等),不同平台和技术的系统可以相互通信。
  4. 分布式:服务可以部署在不同的物理位置和网络环境中,通过网络进行通信。
  5. 灵活性:可以根据需求动态组合和配置服务,适应不同的业务需求。
  6. 可维护性:独立的服务可以单独开发、测试和部署,便于系统的维护和升级。

        作为SOA基于服务的软件架构,通讯方式自然选择网络层的协议进行通信比较好,这样方便进行SOA的动态配置,支持服务的动态发现和调用,因此一般选SOMEIP作为SOA的通讯方式。

        SOME/IP是一种基于IP的通信协议,实现分布式服务之间的通信和数据传输。其特点是:

  1. 标准化:SOME/IP是为汽车电子系统设计的标准化通信协议,支持服务发现、方法调用和事件通知,符合SOA的标准化要求。

  2. 高性能:SOME/IP设计用于满足汽车电子系统的实时性和高带宽需求,具有高效的数据传输能力。

  3. 服务导向:SOME/IP支持服务导向的通信模式,便于实现SOA架构中的服务发现和动态配置。

  4. 可靠性:SOME/IP具有强大的错误处理和恢复机制,确保通信的可靠性和稳定性。

  5. 扩展性:基于IP协议的SOME/IP可以方便地扩展和集成到不同的网络和系统中,适应不断变化的业务需求。

2. SOME/IP的原理和组件

        SOME/IP作为一种应用层协议,运行在TCP/IP协议栈之上,SOME/IP协议定义了服务发现、方法调用和事件通知等机制,传输层使用TCP或UDP协议进行数据传输。

        

工作流程如下:

        1. 服务注册,服务端在启动时,通过广播或多播方式向网络中的其他设备宣布自己的服务,并注册到服务发现模块。

        2. 服务发现,客户端在需要使用某个服务时,通过广播或多播方式发送服务发现请求,寻找可用的服务。      

  3. 方法调用,客户端发现服务后,发送请求消息,调用服务端提供的方法。服务端接收到请求后,处理请求并返回响应消息。

        4. 事件通知,服务端可以发布事件,通知客户端某些状态或数据的变化。客户端订阅特定服务的事件,接收事件消息。

        5. 高效的数据封装,SOME/IP定义了消息的封装格式,包括消息头和消息负载,确保数据能够高效传输。消息头包含服务ID、实例ID、方法ID等信息,便于消息的解析和处理。

3. 实例代码

        以下是一个简单的SOME/IP客户端和服务端示例,展示了基本的服务发现、方法调用和事件通知过程。

        在SOME/IP中,IP地址和端口通常通过配置文件进行管理。以下是一个示例配置文件vsomeip.json,其中包含了IP地址和端口的设置:        

{"unicast": "192.168.0.100",  // 本地IP地址"netmask": "255.255.255.0",  // 子网掩码"gateway": "192.168.0.1",    // 网关"applications": [{"name": "Service",  // 应用名称"id": "0x1234",     // 应用ID"instance": "0x5678",  // 实例ID"service": "0x4321",  // 服务ID"eventgroups": [{"group_id": "0x00",  // 事件组ID"events": [{"event_id": "0x0421"  // 事件ID}]}],"port": 30490,  // 服务端口"protocol": "UDP"  // 传输协议}],"routing": {"protocol": "UDP",  // 路由协议"port": 30509,  // 路由端口"routing_manager": true  // 是否启用路由管理器}
}

        服务端代码:

#include <vsomeip/vsomeip.hpp>
#include <iostream>class SomeIPServer {
public:SomeIPServer() : app_(vsomeip::runtime::get()->create_application("Service")) {}bool init() {app_->init("vsomeip.json");  // 初始化时加载配置文件app_->register_message_handler(SERVICE_ID, INSTANCE_ID, METHOD_ID,std::bind(&SomeIPServer::on_message, this, std::placeholders::_1));app_->register_state_handler(std::bind(&SomeIPServer::on_state, this, std::placeholders::_1));app_->offer_service(SERVICE_ID, INSTANCE_ID);return true;}void start() {app_->start();}private:void on_message(const std::shared_ptr<vsomeip::message> &msg) {std::shared_ptr<vsomeip::payload> payload = msg->get_payload();std::string message(payload->get_data(), payload->get_data() + payload->get_length());std::cout << "Received message: " << message << std::endl;// Sending a responsestd::shared_ptr<vsomeip::message> response = vsomeip::runtime::get()->create_response(msg);std::string response_message = "Hello from the server!";std::shared_ptr<vsomeip::payload> response_payload = vsomeip::runtime::get()->create_payload();response_payload->set_data(reinterpret_cast<const vsomeip::byte_t *>(response_message.data()), response_message.size());response->set_payload(response_payload);app_->send(response);}void on_state(vsomeip::state_type_e state) {if (state == vsomeip::state_type_e::ST_REGISTERED) {app_->offer_service(SERVICE_ID, INSTANCE_ID);}}std::shared_ptr<vsomeip::application> app_;const vsomeip::service_t SERVICE_ID = 0x4321;const vsomeip::instance_t INSTANCE_ID = 0x5678;const vsomeip::method_t METHOD_ID = 0x0421;
};int main() {SomeIPServer server;if (server.init()) {server.start();}return 0;
}

        客户端代码实例:

#include <vsomeip/vsomeip.hpp>
#include <iostream>class SomeIPClient {
public:SomeIPClient() : app_(vsomeip::runtime::get()->create_application("Client")) {}bool init() {app_->init("vsomeip.json");  // 初始化时加载配置文件app_->register_message_handler(SERVICE_ID, INSTANCE_ID, METHOD_ID,std::bind(&SomeIPClient::on_message, this, std::placeholders::_1));app_->register_state_handler(std::bind(&SomeIPClient::on_state, this, std::placeholders::_1));return true;}void start() {app_->start();}void send_request() {std::shared_ptr<vsomeip::message> request = vsomeip::runtime::get()->create_request();request->set_service(SERVICE_ID);request->set_instance(INSTANCE_ID);request->set_method(METHOD_ID);std::string request_message = "Hello from the client!";std::shared_ptr<vsomeip::payload> request_payload = vsomeip::runtime::get()->create_payload();request_payload->set_data(reinterpret_cast<const vsomeip::byte_t *>(request_message.data()), request_message.size());request->set_payload(request_payload);app_->send(request);}private:void on_message(const std::shared_ptr<vsomeip::message> &msg) {std::shared_ptr<vsomeip::payload> payload = msg->get_payload();std::string message(payload->get_data(), payload->get_data() + payload->get_length());std::cout << "Received response: " << message << std::endl;}void on_state(vsomeip::state_type_e state) {if (state == vsomeip::state_type_e::ST_REGISTERED) {send_request();}}std::shared_ptr<vsomeip::application> app_;const vsomeip::service_t SERVICE_ID = 0x4321;const vsomeip::instance_t INSTANCE_ID = 0x5678;const vsomeip::method_t METHOD_ID = 0x0421;
};int main() {SomeIPClient client;if (client.init()) {client.start();}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/384110.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

音视频入门基础:H.264专题(16)——FFmpeg源码中,判断某文件是否为H.264裸流文件的实现

音视频入门基础&#xff1a;H.264专题系列文章&#xff1a; 音视频入门基础&#xff1a;H.264专题&#xff08;1&#xff09;——H.264官方文档下载 音视频入门基础&#xff1a;H.264专题&#xff08;2&#xff09;——使用FFmpeg命令生成H.264裸流文件 音视频入门基础&…

Redis (常用数据结构和命令)

目录 简介 概述 特点 数据结构 常用命令 通用命令 keys del exists expire 与 ttl String 命令 SET 和GET: MSET和MGET INCR和INCRBY和DECY SETNX SETEX Redis 命令 Key 的层级结构 key层级关系 &#xff1a; Hash命令 HSET和HGET HMSET和HMGET HGETALL H…

免杀笔记 -->API的整理Shellcode加密(过DeFender)

最近更新频率明显下降我懒&#xff0c;那么今天就来记录一下我们的一些常用的API的整理以及ShellCode的加密。 1.WinAPI整理 问我为什么要整理&#xff1f; 就是用起来的时候要左翻右翻 &#xff1a;&#xff1a; 烦死了 1.VirtualAlloc VirtualAlloc(NULL,sizeof(buf),MEM_…

FastAPI(七十五)实战开发《在线课程学习系统》接口开发-- 创建课程

源码见&#xff1a;"fastapi_study_road-learning_system_online_courses: fastapi框架实战之--在线课程学习系统" 上次我们分享了&#xff0c;FastAPI&#xff08;七十四&#xff09;实战开发《在线课程学习系统》接口开发-- 删除留言 从本篇文章开始&#xff0c;…

Potree在web端显示大型点云模型文件

一、克隆项目代码&#xff08;准备好上网工具&#xff0c;得先有node.js npm 环境&#xff09; git clone https://github.com/potree/potree.git二、依赖安装&#xff08;换淘宝镜像能快一些&#xff09; cd potree npm install三、运行 npm start四、使用样例 打开浏览器…

python黑马笔记

运算符&#xff1a; 算术运算符&#xff1a; 加 - 减 * 乘 / 除 // 整除 % 取余 ** 求平方 除法计算得出的结果都是小数 赋值运算符&#xff1a; 标准赋值&#xff1a; 复合赋值&#xff1a; 、 - 、 * 、 / 、// 、 ** 字符串&#xff1a; 字符串拓展内容&#xf…

Vue 3 实现左侧列表点击跳转滚动到右侧对应区域的功能

使用 Vue 3 实现左侧列表点击跳转到右侧对应区域的功能 1. 引言 在这篇博客中&#xff0c;我们将展示如何使用 Vue 3 实现一个简单的页面布局&#xff0c;其中左侧是一个列表&#xff0c;点击列表项时&#xff0c;右侧会平滑滚动到对应的内容区域。这种布局在很多应用场景中都…

力扣高频SQL 50 题(基础版)第三题

文章目录 力扣高频SQL 50 题&#xff08;基础版&#xff09;第三题1148.文章浏览题目说明思路分析实现过程准备数据实现方式结果截图 力扣高频SQL 50 题&#xff08;基础版&#xff09;第三题 1148.文章浏览 题目说明 Views 表&#xff1a; ---------------------- | Colu…

Internet Download Manager(IDM)2024中文版本有哪些新功能?6.42版本功能介绍

1. Internet Download Manager&#xff08;IDM&#xff09;是一款功能强大的下载管理器&#xff0c;支持所有流行的浏览器&#xff0c;并可提升下载速度高达5倍。 2. IDM具有智能下载逻辑加速器&#xff0c;可以设置文件下载优先级、分块下载等&#xff0c;提高下载效率。 IDM…

数据结构(5.3_4)——线索二叉树的概念

普通二叉树找某结点前驱和后继的方法 中序线索二叉树 n个结点的二叉树&#xff0c;有n1个空链域!可用来记录前驱&#xff0c;后继的信息 中序线索二叉树的存储结构 //线索二叉树结点 typedef struct ThreadNode {ElemType data;struct BiTNode* lchild, * rchild;int ltag,…

Facebook的隐私之战:用户数据保护的挑战与未来

在数字化时代&#xff0c;隐私问题成为了公众关注的焦点&#xff0c;而作为全球最大的社交网络平台之一&#xff0c;Facebook&#xff08;现已更名为Meta&#xff09;在用户数据保护方面面临着巨大的挑战。从数据泄露到隐私政策的变化&#xff0c;Facebook的隐私之战不仅关乎其…

前端三大主流框架Vue React Angular有何不同?

前端主流框架&#xff0c;Vue React Angular&#xff0c;大家可能都经常在使用&#xff0c;Vue React&#xff0c;国内用的较多&#xff0c;Angualr相对用的少一点。但是大家有思考过这三大框架的不同吗&#xff1f; 一、项目的选型上 中小型项目&#xff1a;Vue2、React居多…

规范:前后端接口规范

1、前言 随着互联网的高速发展&#xff0c;前端页面的展示、交互体验越来越灵活、炫丽&#xff0c;响应体验也要求越来越高&#xff0c;后端服务的高并发、高可用、高性能、高扩展等特性的要求也愈加苛刻&#xff0c;从而导致前后端研发各自专注于自己擅长的领域深耕细作。 然…

SpringMVC基础

SpringMVC ssm&#xff1a;mybatisSpringSpringMVC MVC三层架构 1、什么是MVC MVC是模型&#xff08;Model&#xff09;、视图&#xff08;View&#xff09;、控制器&#xff08;Controller&#xff09;的简写&#xff0c;是一种软件设计规范 是将业务逻辑、数据、显示分离…

应用层自定义协议以及序列化和反序列化

文章目录 应用层自定义协议以及序列化和反序列化1、应用层自定义协议1.1、应用层1.2、协议 2、序列化和反序列化3、TCP 为什么支持全双工4、jsoncpp基础4.1、序列化4.2、反序列化 5、实现网络版计算器6、手写序列化和反序列化 应用层自定义协议以及序列化和反序列化 1、应用层…

Anconda 快速常用命令简洁版

目的&#xff1a;简单清楚的使用基本的conda 命令 可能需求 查看项目中的虚拟环境及依赖是否满足需求操作新环境来满足项目或者论文的实现 Anconda 常用命令 conda 查看基础命令1. 进入Anaconda 环境2. 查看版本3.查看有哪些虚拟环境4.激活虚拟环境5. 进入虚拟环境查看6. 退出…

基于STM32瑞士军刀--【FreeRTOS开发】学习笔记(二)|| 堆 / 栈

堆和栈 1. 堆 堆就是空闲的一块内存&#xff0c;可以通过malloc申请一小块内存&#xff0c;用完之后使用再free释放回去。管理堆需要用到链表操作。 比如需要分配100字节&#xff0c;实际所占108字节&#xff0c;因为为了方便后期的free&#xff0c;这一小块需要有个头部记录…

IDEA缓存和索引

IDEA缓存和索引 —2020年06月10日 IntelliJ IDEA首次加载项目的时候。都会创建索引&#xff0c;而创建索引的时间根项目的文件多少成正比。 IntelliJ IDEA的缓存和索引主要是用来加快文件查询&#xff0c;从而加快各种查找、代码提示等操作的速度。 某些特殊情况下&#xf…

Go语言入门

目录 前言——碎碎念 环境安装 配置环境变量 变量的定义 数据类型 数字型 字符与字符串 数据类型转换 运算符 算术运算符 关系运算符 逻辑运算符 位运算符&#xff08;二进制&#xff09; 赋值运算符 其他运算符&#xff08;指针&#xff09; 键盘的输入与输出…

使用 Python创建照片文件复制和压缩工具

在这篇博客中&#xff0c;我们将探索如何使用 wxPython 创建一个 GUI 工具&#xff0c;用于选择文件夹中的照片文件、显示预览、选择并复制文件到指定目录&#xff0c;以及将选中的照片压缩到一个 ZIP 文件中。这个工具不仅功能强大&#xff0c;而且提供了用户友好的界面。 C:\…