【ROS教程】服務通信

来源:https://www.cnblogs.com/UnderTurrets/p/18386484
-Advertisement-
Play Games

@目錄1.流程2.自定義請求和響應的數據2.1 std_msgs內置類型2.2 編寫.srv文件2.3 修改package.xml文件2.4 修改CMakeLists.txt文件2.4.1 修改find_package指令2.4.2 添加add_message_files指令2.4.3 添加gene ...


@

目錄


1.流程

服務通信也是ROS中一種極其常用的通信模式,服務通信是基於請求響應模式的,是一種應答機制。也即: 一個節點A向另一個節點B發送請求,B接收處理請求並產生響應結果返回給A。在ROS中,實現服務通信只需要如下幾步:

  1. 確定客戶端發送的請求的數據類型和服務端響應的數據類型需要自定義.srv文件,修改好CMakeLists.txt文件和package.xml文件並重編譯
  2. 編寫發佈方和訂閱方的cpp文件,修改好CMakeLists.txt文件並重編譯
  3. 分別啟動發佈方節點和訂閱方節點,必須先啟動服務端再啟動客戶端

2.自定義請求和響應的數據

2.1 std_msgs內置類型

  • 內置類型與 C++ 和 Python 中的對應關係:
Primitive Type C++ Python
bool uint8_t bool
int8 int8_t int
uint8 uint8_t int
int16 int16_t int
uint16 uint16_t int
int32 uint32_t int
uint64 uint64_t long int
float32 float float
float64 double float
string std::string str bytes
time ros::Time rospy.Time
duration ros::Duration rospy.Duration
  • 內置類型的數組與 C++ 和 Python 中的對應關係:
Primitive Type C++ Python
variable-length std::vector tuple
fixed-length boost::array<T, length>或std::vector tuple

2.2 編寫.srv文件

示例如下:

#文件名AddInt.srv
# 客戶端請求時發送的兩個數字
int32 num1
int32 num2
---
# 伺服器響應發送的數據
int32 sum
  • 中間的三橫用以區分請求數據和響應數據

2.3 修改package.xml文件

  • 查看是否存在如下編譯依賴
<build_depend>message_generation</build_depend>
  • 查看是否存在如下執行依賴
<exec_depend>message_generation</exec_depend>

2.4 修改CMakeLists.txt文件

2.4.1 修改find_package指令

# 需要加入 message_generation,必須有 std_msgs
find_package(catkin REQUIRED COMPONENTS
  roscpp
  rospy
  std_msgs
  message_generation
)

2.4.2 添加add_message_files指令

## 配置 srv 源文件
add_service_files(
  FILES
  AddInt.srv
)

2.4.3 添加generate_messages指令

generate_messages(
  DEPENDENCIES
  std_msgs
)

其中,add_service_files指令必須要在generate_messages指令的前面,然後在工作空間目錄編譯即可。

2.5 查看頭文件

經過以上幾步,在${workspace}/devel/include/${package}/目錄下應該會出現頭文件,如圖:

在這裡插入圖片描述

  • 如果沒有出現,是無法進行接下來的步驟的。這時,只需要把${workspace}/目錄下的build目錄和devel目錄全部刪除,然後重新編譯即可。
rm -rf build/
rm -rf devel/
catkin_make

3.編寫cpp文件

3.1 功能包目錄文件樹

在這裡插入圖片描述

3.2 修改CMakeLists.txt文件

3.2.1 添加add_executable指令

add_executable(server src/server.cpp)
add_executable(client src/client.cpp)

3.2.2 添加add_dependencies指令

add_dependencies(server ${PROJECT_NAME}_gencpp)
add_dependencies(client ${PROJECT_NAME}_gencpp)
target_link_libraries(server
        ${catkin_LIBRARIES}
        )
target_link_libraries(client
        ${catkin_LIBRARIES}
        )

3.3 服務端cpp

示例如下:

/*
    需求:
        編寫兩個節點實現服務通信,客戶端節點需要提交兩個整數到伺服器
        伺服器需要解析客戶端提交的數據,相加後,將結果響應回客戶端,
        客戶端再解析

    伺服器實現:
        1.包含頭文件
        2.初始化 ROS 節點
        3.創建 ROS 句柄
        4.創建 服務 對象
        5.回調函數處理請求並產生響應
        6.由於請求有多個,需要調用 ros::spin()

*/
// 1.包含頭文件
#include "ros/ros.h"
#include "serve/AddInt.h"

// bool 返回值由於標誌是否處理成功
bool reponse(serve::AddInt::Request& req,
           serve::AddInt::Response& resp){
    int num1 = req.num1;
    int num2 = req.num2;

    ROS_INFO("伺服器接收到的請求數據為:num1 = %d, num2 = %d",num1, num2);

    //邏輯處理
    if (num1 < 0 || num2 < 0)
    {
        ROS_ERROR("提交的數據異常:數據不可以為負數");
        return false;
    }

    //如果沒有異常,那麼相加並將結果賦值給 resp
    resp.sum = num1 + num2;
    return true;


}

int main(int argc, char *argv[])
{
    //設置編碼
    setlocale(LC_ALL,"");
    // 2.初始化 ROS 節點
    ros::init(argc,argv,"server");
    // 3.創建 ROS 句柄
    ros::NodeHandle nh;
    // 4.創建 服務 對象,回調函數的返回值必須是布爾類型
    ros::ServiceServer server = nh.advertiseService("AddInt",reponse);
    ROS_INFO("服務已經啟動....");
    //     5.回調函數處理請求並產生響應
    //     6.由於請求有多個,需要調用 ros::spin()
    ros::spin();
    return 0;
}
  • 創建服務對象時的回調函數的返回值必須是布爾類型

3.4 客戶端cpp

示例如下:

/*
    需求:
        編寫兩個節點實現服務通信,客戶端節點需要提交兩個整數到伺服器
        伺服器需要解析客戶端提交的數據,相加後,將結果響應回客戶端,
        客戶端再解析

    伺服器實現:
        1.包含頭文件
        2.初始化 ROS 節點
        3.創建 ROS 句柄
        4.創建 客戶端 對象
        5.請求服務,接收響應

*/
// 1.包含頭文件
#include "ros/ros.h"
#include "serve/AddInt.h"

int main(int argc, char *argv[])
{
    //設置編碼
    setlocale(LC_ALL,"");

    // 調用時動態傳值,如果通過 launch 的 args 傳參,需要傳遞的參數個數 +3
    if (argc != 3)
        // if (argc != 5)//launch 傳參(0-文件路徑 1傳入的參數 2傳入的參數 3節點名稱 4日誌路徑)
    {
        ROS_ERROR("請提交兩個整數");
        return 1;
    }


    // 2.初始化 ROS 節點
    ros::init(argc,argv,"client");
    // 3.創建 ROS 句柄
    ros::NodeHandle nh;
    // 4.創建 客戶端 對象
    ros::ServiceClient client = nh.serviceClient<serve::AddInt>("AddInt");
    //等待服務啟動成功
    //方式1
    ros::service::waitForService("AddInt");
    //方式2
    // client.waitForExistence();
    // 5.組織請求數據
    serve::AddInt ai;
    ai.request.num1 = atoi(argv[1]);
    ai.request.num2 = atoi(argv[2]);
    // 6.發送請求,返回 bool 值,標記是否成功
    bool flag = client.call(ai);
    // 7.處理響應
    if (flag)
    {
        ROS_INFO("請求正常處理,響應結果:%d",ai.response.sum);
    }
    else
    {
        ROS_ERROR("請求處理失敗....");
        return 1;
    }

    return 0;
}

4.效果

在這裡插入圖片描述

在這裡插入圖片描述

在這裡插入圖片描述

在這裡插入圖片描述

本文由博客一文多發平臺 OpenWrite 發佈!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在TS開發中,經常會遇到後臺數據欄位比較多的情況,這時候需要一個個複製欄位然後給他手動配置數據類型來完成我們的TS類型定義,相當麻煩。有什麼快速的方法呢,我就目前遇到的兩種情況分別寫了JS腳本來處理後臺數據,直接生成我們需要的數據格式。 腳本編寫 1. 處理數據字典中的數據 一般數據字典表裡的數據可 ...
  • 此“華仔”,不是彼“華仔”,你懂的! 先來了個截圖 緊跟著,實現步驟也來了 1. 安裝 Node.js,終端運行 npm create vue@latest,項目名 vue-to-do,後面的選項全選 No; 2. cd vue-to-do 進入項目目錄,npm install 安裝依賴,npm r ...
  • title: 使用 Nuxt 的 showError 顯示全屏錯誤頁面 date: 2024/8/26 updated: 2024/8/26 author: cmdragon excerpt: 摘要:本文介紹Nuxt.js中的showError方法用於顯示全屏錯誤頁面,包括其參數類型及使用方式,並演 ...
  • 信息系統設計速查表需求收集功能性需求:通過用戶訪談、問卷調查和市場分析來確定系統必須實現的功能。 非功能性需求:定義性能基準(如響應時間)、安全性要求(如數據加密標準,傳輸加密)、可靠性標準(如系統的正常運行時間, MTTR)。 用戶故事:使用敏捷方法論,如Scrum,來創建和優先順序排序用戶故事。 ...
  • @目錄1.流程2.自定義發佈數據2.1 std_msgs內置類型2.2 編寫.msg文件2.3 修改package.xml文件2.3.1 完整的package.xml文件2.4 修改CMakeLists.txt文件2.4.1 修改find_package指令2.4.2 添加add_message_f ...
  • IOC與DI的理解 1.1、IoC是什麼 Ioc—Inversion of Control,即“控制反轉”,不是什麼技術,而是一種設計思想。在Java開發中,Ioc意味著將你設計好的對象交給容器控制,而不是傳統的在你的對象內部直接控制。如何理解好Ioc呢?理解好Ioc的關鍵是要明確“誰控制誰,控制什 ...
  • @目錄1.launch文件有哪些標簽2.node標簽2.1 必選屬性2.2 可選屬性2.3 可選子級標簽3.include標簽3.1 必選屬性3.2 可選屬性3.3 可選子級標簽4.remap標簽4.1 必選屬性5.param標簽5.1 必選屬性5.2 可選屬性6.rosparam標簽6.1 必選屬 ...
  • @目錄1.工作空間目錄2.載入環境變數3.打開CLion4.配置CLion5.編譯和調試軟體包 1.工作空間目錄 我們的一個工作空間目錄應該是這樣的 2.載入環境變數 先進入工作空間再載入 source ./devel/setup.bash 3.打開CLion 一定要在第二步的同一個終端下 clio ...
一周排行
    -Advertisement-
    Play Games
  • 前言 推薦一款基於.NET 8、WPF、Prism.DryIoc、MVVM設計模式、Blazor以及MySQL資料庫構建的企業級工作流系統的WPF客戶端框架-AIStudio.Wpf.AClient 6.0。 項目介紹 框架採用了 Prism 框架來實現 MVVM 模式,不僅簡化了 MVVM 的典型 ...
  • 先看一下效果吧: 我們直接通過改造一下原版的TreeView來實現上面這個效果 我們先創建一個普通的TreeView 代碼很簡單: <TreeView> <TreeViewItem Header="人事部"/> <TreeViewItem Header="技術部"> <TreeViewItem He ...
  • 1. 生成式 AI 簡介 https://imp.i384100.net/LXYmq3 2. Python 語言 https://imp.i384100.net/5gmXXo 3. 統計和 R https://youtu.be/ANMuuq502rE?si=hw9GT6JVzMhRvBbF 4. 數 ...
  • 本文為大家介紹下.NET解壓/壓縮zip文件。雖然解壓縮不是啥核心技術,但壓縮性能以及進度處理還是需要關註下,針對使用較多的zip開源組件驗證,給大家提供個技術選型參考 之前在《.NET WebSocket高併發通信阻塞問題 - 唐宋元明清2188 - 博客園 (cnblogs.com)》講過,團隊 ...
  • 之前寫過兩篇關於Roslyn源生成器生成源代碼的用例,今天使用Roslyn的代碼修複器CodeFixProvider實現一個cs文件頭部註釋的功能, 代碼修複器會同時涉及到CodeFixProvider和DiagnosticAnalyzer, 實現FileHeaderAnalyzer 首先我們知道修 ...
  • 在軟體行業,經常會聽到一句話“文不如表,表不如圖”說明瞭圖形在軟體應用中的重要性。同樣在WPF開發中,為了程式美觀或者業務需要,經常會用到各種個樣的圖形。今天以一些簡單的小例子,簡述WPF開發中幾何圖形(Geometry)相關內容,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 在 C# 中使用 RabbitMQ 通過簡訊發送重置後的密碼到用戶的手機號上,你可以按照以下步驟進行 1.安裝 RabbitMQ 客戶端庫 首先,確保你已經安裝了 RabbitMQ 客戶端庫。你可以通過 NuGet 包管理器來安裝: dotnet add package RabbitMQ.Clien ...
  • 1.下載 Protocol Buffers 編譯器(protoc) 前往 Protocol Buffers GitHub Releases 頁面。在 "Assets" 下找到適合您系統的壓縮文件,通常為 protoc-{version}-win32.zip 或 protoc-{version}-wi ...
  • 簡介 在現代微服務架構中,服務發現(Service Discovery)是一項關鍵功能。它允許微服務動態地找到彼此,而無需依賴硬編碼的地址。以前如果你搜 .NET Service Discovery,大概率會搜到一大堆 Eureka,Consul 等的文章。現在微軟為我們帶來了一個官方的包:Micr ...
  • ZY樹洞 前言 ZY樹洞是一個基於.NET Core開發的簡單的評論系統,主要用於大家分享自己心中的感悟、經驗、心得、想法等。 好了,不賣關子了,這個項目其實是上班無聊的時候寫的,為什麼要寫這個項目呢?因為我單純的想吐槽一下工作中的不滿而已。 項目介紹 項目很簡單,主要功能就是提供一個簡單的評論系統 ...