【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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...