第 4 節

四大通信

0瀏覽次數0訪問次數--跳出率--平均停留

通信機制簡介與代碼模板

#include "rclcpp/rclcpp.hpp"

class MyNode: public rclcpp::Node
{
  public:
    MyNode():Node("mynode_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"Hello World!");
    }
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  rclcpp::spin(std::make_shared<MyNode>());

  rclcpp::shutdown();
  return 0;
}

https://snippet-generator.app/

{
"ROS2节点模板(C++)": {
    "prefix": "ros2_node_cpp",
    "body": [
      "#include \"rclcpp/rclcpp.hpp\"",
      "",
      "class MyNode: public rclcpp::Node",
      "{",
      "  public:",
      "    MyNode():Node(\"mynode_node_cpp\")",
      "    {",
      "      RCLCPP_INFO(this->get_logger(),\"Hello World!\");",
      "    }",
      "};",
      "",
      "int main(int argc, char ** argv)",
      "{",
      "  rclcpp::init(argc,argv);",
      "",
      "  rclcpp::spin(std::make_shared<MyNode>());",
      "",
      "  rclcpp::shutdown();",
      "  return 0;",
      "}"
    ],
    "description": "ROS2节点模板(C++)"
  }
}


第一個窗口是服務端,第二個窗口是客戶端。

通信至少要涉及兩方,只是一個人算不上通信。

面向接口,話題是一致的,數據載體也是一致的,就可以無縫對接

話題通信:只能單向傳輸數據

服務通信:雙向通信,可以互為客戶端和服務端,客戶端給服務端發數據,服務端給客戶端響應

動作通信:和服務通信很像,有服務端給客戶端發的最終響應,但是中間也會連續發送反饋給客戶端。

參數服務:參數服務是基於服務通信的,參數客戶先發送一個請求,然後從參數服務的數據池裡拿走數據。也可以更改數據池裡的東西,但是不能刪除。

參數通信不用自己定義接口文件,系統會自己弄接口文件,但是開發者是看不到該文件的,該文件被封裝了。

我們操作的數據被封裝成參數對象了。


ros2 pkg create + 功能包名(可以寫在前面或者) + --build-type(構建類型) + ament_cmake / ament_python + --dependencies(依賴) + rclcpp(ROS2的CPP客戶端) + --node-name(節點名) + 節點名

在src裡創建功能包

話題通信_理論

在ROS1裡,node和node之間通信需要經過master,每個傳輸數據的node都需要在master裡註冊相關數據,master再將信息進行匹配。

一個publisher發佈數據,兩個subscriber都會接收到數據。

話題通信_實驗1(C++)

ros2 pkg create cpp01_topic --build-type ament_cmake --dependencies rclcpp std_msgs base_interfaces_demo

依賴還需要std_msgs,base_interfaces_demo(這裡面存放了我們自己定義的所需的接口)

ros2 pkg create + 功能包名(可以寫在前面或者最後面) + --build-type(構建類型) + ament_cmake / ament_python + --dependencies(依賴) + rclcpp(ROS2的CPP客戶端) + --node-name(節點名) + 節點名



發佈方

定時器是用來控制發送頻率的,定時器裡還會執行一個回調函數timer_callback。

count_是計數器,每執行一次這個回調函數,count_就++;

spin函數是,程序一旦執行到這裡,就返回到上面,返回到上面是為了調用回調函數,如果沒有這個spin函數,那麼我們這個回調函數是不會被執行的。

以後只要我們創建完節點類對象指針,就要把該指針傳入spin()函數里。


#include "rclcpp/rclcpp.hpp"

class Talker: public rclcpp::Node
{
  public:
    Talker():Node("talker_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"发布节点创建!");
    }
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  rclcpp::spin(std::make_shared<Talker>());

  rclcpp::shutdown();
  return 0;
}

此時程序處於掛起狀態,會一直運行,因為spin函數。

想結束得按Ctrl+C。

我們想要的類型在std_msgs裡,所以要加頭文件。

create_publisher()第一個入口參數是話題名稱,是一個字符串

create_publisher()第二個入口參數是QOS服務質量有關的,是隊列深度是一串數字,暫時可以先填10或者20等。

也就是當網絡質量不好的時候,消息發不出去了,我們可以將數據先存到隊列裡,假設填10,最多就可以存10個數,當網絡恢復了,我們就從隊列裡取數據,將其發出。

其他入口參數有默認值,可以暫時先不管。

返回值是一個publisher的指針。

創建定時器,這個函數有模板,但是模板有默認值可以不設置,

然後三個入口參數,

第一個入口參數是持續時間,也就是週期;

第二個入口參數是回調函數;

第三個入口參數有默認值,先不管。

使用該命名空間的優勢是,在第一個入口參數,可以直接填時間+單位。

如果是1s就寫1s,是100ms就填100ms。

該函數還有個返回值,返回值是定時器相關的一個指針。

該函數有多個重載,選擇適合自己的一個函數。

發佈對象得先創建對象。

把count轉化成字符串併發送。

因為它是一個std::string類型的,我們要轉化成c風格的字符串。

儘量在構造函數的時候,給count賦初值0.

#include "rclcpp/rclcpp.hpp"
#include "std_msgs/msg/string.hpp"

using namespace std::chrono_literals;

class Talker: public rclcpp::Node
{
  public:
    Talker():Node("talker_node_cpp"),count(0)
    {
      RCLCPP_INFO(this->get_logger(),"发布节点创建!");
      publisher_ = this->create_publisher<std_msgs::msg::String>("chatter",10);
      timer_ = this->create_wall_timer(1s,std::bind(&Talker::on_timer,this));
    }
  private:
    void on_timer()
    {
      auto message = std_msgs::msg::String();
      message.data = "hello world!" + std::to_string(count++);
      RCLCPP_INFO(this->get_logger(),"发布方发布的消息:%s",message.data.c_str());
      publisher_->publish(message);
    }
    rclcpp::Publisher<std_msgs::msg::String>::SharedPtr publisher_;
    rclcpp::TimerBase::SharedPtr timer_;
    size_t count;
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  rclcpp::spin(std::make_shared<Talker>());

  rclcpp::shutdown();
  return 0;
}

但是不一定消息真的發佈出去了。

驗證方法:

ros2 topic echo /xxx

這樣才能確定消息被髮到chatter話題上了。


#include "rclcpp/rclcpp.hpp"

class Listener: public rclcpp::Node
{
public:
    Listener():Node("listener_node_cpp")
    {
        RCLCPP_INFO(this->get_logger(),"订阅方创建!");
    }
};

int main(int argc, char * argv[])
{
    rclcpp::init(argc,argv);

    rclcpp::spin(std::make_shared<Listener>());

    rclcpp::shutdown();
    return 0;
}

編譯之前別忘了編輯配置文件

依賴包已經自動生成了,不用管。

主要改這三大部分

一共有5個入口參數,後面兩個入口參數有默認值。

第一個入口參數是話題名稱,要保證和發佈方一致;

第二個入口參數是QoS,就是服務質量管理,隊列深度,10或者20暫時隨便設置,可以看看發佈方那邊的QoS的解釋;

第三個入口參數是回調函數,一旦接收到數據,就觸發該回調函數。

返回值是一個訂閱對象的指針。

std::placeholders::_1這個是佔位符,_1是指一個。這個地方本應該填入消息。

#include "rclcpp/rclcpp.hpp"
#include "std_msgs/msg/string.hpp"

class Listener: public rclcpp::Node
{
public:
    Listener():Node("listener_node_cpp")
    {
        RCLCPP_INFO(this->get_logger(),"订阅方创建!");
        subscription_ = this->create_subscription<std_msgs::msg::String>("chatter",10,std::bind(&Listener::do_callback,this,std::placeholders::_1));

    }
private:
    void do_callback(const std_msgs::msg::String &msg)
    {
        RCLCPP_INFO(this->get_logger(),"订阅到的消息是:%s",msg.data.c_str());
    }
    rclcpp::Subscription<std_msgs::msg::String>::SharedPtr subscription_;
};

int main(int argc, char * argv[])
{
    rclcpp::init(argc,argv);

    rclcpp::spin(std::make_shared<Listener>());

    rclcpp::shutdown();
    return 0;
}

話題通信_自定義接口信息

構建依賴

執行依賴

當前功能包所屬的功能包組

find_package是要把構建依賴傳遞過來

然後還要指定當前被構建的接口文件的路徑(通過這個設置,就可以把.msg轉化成對應的C++和Python代碼了)

文件名可以自定義,但是首字母必須大寫

寫完之後test_depend報錯了

刪掉即可

編譯依賴是rosidl開頭的,我們通過grep查詢一下

ros2 pkg list | grep -i rosidl

我們用的是這一個,直接複製過來

在list裡這個所屬的功能包組就沒有了,需要自己寫rosidl_interface_packages

這個依賴要和構建依賴一樣

然後我們要為接口文件生成源碼

需要使用rosidl_generate_interfaces函數

colcon build --packages-select base_interfaces_demo

會在install空間下生成student.hpp代碼

以上.msg生成C++的

然後這個是.msg生成的Python的源碼

也可以通過這個方式來檢測是否編譯正常

interface是接口

話題通信_實驗2(C++)

新建完源文件之後,要配置CMakeLists

將最基本的框架直接複製過來

然後替換類的名稱

#include "rclcpp/rclcpp.hpp"

class TalkerStu: public rclcpp::Node
{
  public:
    TalkerStu():Node("talkerstu_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"发布节点创建!");
    }
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  rclcpp::spin(std::make_shared<TalkerStu>());

  rclcpp::shutdown();
  return 0;
}

為了編碼規範,把talkerstu_node_cpp改成小寫

同樣的,訂閱方也是需要這樣操作

#include "rclcpp/rclcpp.hpp"

class ListenerStu: public rclcpp::Node
{
public:
    ListenerStu():Node("listenerstu_node_cpp")
    {
        RCLCPP_INFO(this->get_logger(),"订阅方创建!");
    }
};

int main(int argc, char * argv[])
{
    rclcpp::init(argc,argv);

    rclcpp::spin(std::make_shared<ListenerStu>());

    rclcpp::shutdown();
    return 0;
}

{
  "configurations": [
    {
      "browse": {
        "databaseFilename": "${default}",
        "limitSymbolsToIncludedHeaders": false
      },
      "includePath": [
        "/opt/ros/humble/include/**",
        "/home/tungchiahui/mysource/ros2src/3.ws01_plumbing/src/base_interfaces_demo/include/**",
        "/usr/include/**",
        "${workspaceFolder}/",
**        "${workspaceFolder}/install/base_interfaces_demo/include/**"
      ],
      "name": "ROS",
      "intelliSenseMode": "gcc-x64",
      "compilerPath": "/usr/bin/gcc",
      "cStandard": "gnu11",
      "cppStandard": "c++14"
    }
  ],
  "version": 4
}


不要忘記字符串轉成C風格的。

#include "rclcpp/rclcpp.hpp"
#include "base_interfaces_demo/msg/student.hpp"

using namespace std::chrono_literals;

class TalkerStu: public rclcpp::Node
{
public:
    TalkerStu():Node("talkerstu_node_cpp"),age(0)
    {
      RCLCPP_INFO(this->get_logger(),"发布节点创建!");
      publisher_ = this->create_publisher<base_interfaces_demo::msg::Student>("chatter_stu",10);
      timer_ = this->create_wall_timer(500ms,std::bind(&TalkerStu::on_timer_callback,this));
    }
private:
    void on_timer_callback()
    {
        auto stu = base_interfaces_demo::msg::Student();
        stu.name = "葫芦娃";
        stu.age = age;
        stu.height = 2.20f;
        age++;
        publisher_->publish(stu);
        RCLCPP_INFO(this->get_logger(),"发布的消息:(%s,%d,%.2f)",stu.name.c_str(),stu.age,stu.height);
    }
    rclcpp::Publisher<base_interfaces_demo::msg::Student>::SharedPtr publisher_;
    rclcpp::TimerBase::SharedPtr timer_;
    int32_t age;
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  rclcpp::spin(std::make_shared<TalkerStu>());

  rclcpp::shutdown();
  return 0;
}

雖然發佈方可以打印日誌,但是不代表信息被正常發出去了。

這樣檢驗才是真能確定數據被髮送出去了。


#include "rclcpp/rclcpp.hpp"
#include "base_interfaces_demo/msg/student.hpp"

using base_interfaces_demo::msg::Student;

class ListenerStu: public rclcpp::Node
{
public:
    ListenerStu():Node("listenerstu_node_cpp")
    {
        RCLCPP_INFO(this->get_logger(),"订阅方创建!");
        Subscription_ = this->create_subscription<Student>("chatter_stu",10,std::bind(&ListenerStu::do_callback,this,std::placeholders::1));
    }
private:
    void do_callback(const Student &stu)
    {
        RCLCPP_INFO(this->get_logger(),"订阅的学生信息:name=%s,age=%d,height=%.2f",stu.name.c_str(),stu.age,stu.height);
    }
_    rclcpp::Subscription<Student>::SharedPtr Subscription_;
};

int main(int argc, char * argv[])
{
    rclcpp::init(argc,argv);

    rclcpp::spin(std::make_shared<ListenerStu>());

    rclcpp::shutdown();
    return 0;
}

話題通信_rqt查看計算圖


圖形化工具RQT

服務通信_理論

只能有一個服務端,可以有多個客戶端,每個客戶端都可以向服務端發送請求。(當然可以有多個服務端,但是會出很多邏輯問題,這是極其不合理的,禁止使用)

服務通信_實驗1_服務端實現(C++)

先開服務端,然後從客戶端提交兩個整數,到服務端之後,服務端會解析數據,然後求和,並返回給客戶端。

ros2 pkg create cpp02_service --build-type ament_cmake --dependencies rclcpp base_interfaces_demo --node-name demo01_server

如果之前用過demo_interfaces_demo,那麼一般是不用再配置package.xml了。

記得文件名首字母要大寫!

這是另一個驗證方式


#include "rclcpp/rclcpp.hpp"

class AddIntsServer: public rclcpp::Node
{
  public:
    AddIntsServer():Node("add_ints_server_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"服务端节点创建!");
    }
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  rclcpp::spin(std::make_shared<AddIntsServer>());

  rclcpp::shutdown();
  return 0;
}

服務端是一直要掛起的

客戶端是執行完畢就結束返回到終端的

所以客戶端不用調用spin函數,直接創建對象即可。

#include "rclcpp/rclcpp.hpp"

class AddIntsClient: public rclcpp::Node
{
  public:
    AddIntsClient():Node("add_ints_client_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"客户端节点创建!");
    }
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

//   rclcpp::spin(std::make_shared<AddIntsClient>());
  auto client = std::make_shared<AddIntsClient>();

  rclcpp::shutdown();
  return 0;
}

然後還要編輯配置文件

package.xml現在不用修改


有4個入口參數,但是後兩個有默認值,所以我們只用管前2個。

第一個入口參數就是一個話題名稱,字符串

第二個入口參數是回調函數

返回值是一個service類型的智能指針

#include "rclcpp/rclcpp.hpp"
#include "base_interfaces_demo/srv/add_ints.hpp"

using base_interfaces_demo::srv::AddInts;

class AddIntsServer: public rclcpp::Node
{
  public:
    AddIntsServer():Node("add_ints_server_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"服务端节点创建!");
      server_ = this->create_service<base_interfaces_demo::srv::AddInts>("add_ints",std::bind(&AddIntsServer::add_callback,this,std::placeholders::_1,std::placeholders::2));
    }
  private:
    void add_callback(const AddInts::Request::SharedPtr req,AddInts::Response::SharedPtr res)
    {
      res->sum = req->num1 + (*req).num2;
      RCLCPP_INFO(this->get_logger(),"%d + %d = %d",req->num1,req->num2,res->sum);
    }
_    rclcpp::Service<base_interfaces_demo::srv::AddInts>::SharedPtr server_;
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  rclcpp::spin(std::make_shared<AddIntsServer>());

  rclcpp::shutdown();
  return 0;
}

因為我們的客戶端還沒寫,所以先暫時用ros2 service call這個小工具來查看服務端的情況

ros2 service call /add_ints base_interfaces_demo/srv/AddInts "{'num1': 10,'num2': 30}"

ros2 service call + 話題名 + 接口數據類型 + json代碼(也可以理解成yaml格式的)

此json代碼(yaml格式)格式: "{'第一個數的名': 空格 +對應數值,'第二個數的名': 空格 +對應數的數值}"

服務通信_實驗1_客戶端實現(C++)

運行的時候後面跟了兩個整形數據,

所以這個argc應該是等於3的。

argv[]:接收編譯時的返回的argc的參數

argc是命令行總的參數個數

argv[]是argc個參數,其中第0個參數是程序的全名,以後的參數

必須得保證服務器開著,並且客戶端能夠連接服務器,如果服務器沒開,那麼發送的數據會丟失,但是一般使用服務通信的都是比較重要的信息,一定不要丟失了。

客戶端發送完數據後,會產生一個響應,這裡直接當函數的返回值給返回了。

#include "rclcpp/rclcpp.hpp"

class AddIntsClient: public rclcpp::Node
{
  public:
    AddIntsClient():Node("add_ints_client_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"客户端节点创建!");
    }
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

//   rclcpp::spin(std::make_shared<AddIntsClient>());
  auto client = std::make_shared<AddIntsClient>();

  rclcpp::shutdown();
  return 0;
}

這一段就應該放在節點初始化前面,防止多作一些耗資源的操作再進行判斷。

因為RCLCPP_INFO在節點創建的前面,無法使用類和實例化方式進行get_logger,也就是無法使用this指針和節點智能指針來獲取。

所以我們採用以下方式:

這種方式通過rclcpp裡的get_logger,但是需要給日誌起個名字,放到入口參數裡,我們就叫rclcpp吧。

#include "rclcpp/rclcpp.hpp"

class AddIntsClient: public rclcpp::Node
{
  public:
    AddIntsClient():Node("add_ints_client_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"客户端节点创建!");
    }
};

int main(int argc, char ** argv)
{
  if(argc != 3)
  {
    RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"请提交两个整形数字!");
    return 1;
  }
  rclcpp::init(argc,argv);

  auto client = std::make_shared<AddIntsClient>();

  rclcpp::shutdown();
  return 0;
}

如果我不提交參數,直接回車,然後這是一個異常,主函數返回值不是0

也可以把INFO改成ERROR

#include "rclcpp/rclcpp.hpp"

class AddIntsClient: public rclcpp::Node
{
  public:
    AddIntsClient():Node("add_ints_client_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"客户端节点创建!");
    }
};

int main(int argc, char ** argv)
{
  if(argc != 3)
  {
    RCLCPP_ERROR(rclcpp::get_logger("rclcpp"),"请提交两个整形数字!");
    return 1;
  }
  rclcpp::init(argc,argv);

  auto client = std::make_shared<AddIntsClient>();

  rclcpp::shutdown();
  return 0;
}

一共3個入口參數,

第一個入口參數是話題名稱,是字符串;

第二個入口參數和第三個入口參數有默認值,先不用管;

返回值是客戶端的智能指針。

一旦Ctrl+C關閉,則會瘋狂爆INFO,且程序無法停止。

再按Ctrl+Z可以停止程序進行。

解決上面Ctrl+C的bug:

rclcpp::ok()這個是判斷當前程序是否正常運行,如果正常運行,則返回true,如果不正常運行則返回false,比如按下Ctrl+C就是不正常運行。

當rclcpp::ok() != true的時候,就是ctrl+c按下了。

這樣直接可以讓函數結束。

這時按下ctrl+c會爆很多錯誤

這是因為

this->get_logger()

client->get_logger()

rclcpp::get_logger()

的不同

這個異常和context有關,初始化的時候會創建context對象,相當於是一個容器,可以往容器裡放數據,也可以在容器裡取數據。

當前,如果我們連接失敗的話,打印日誌。

按下ctrl+c會結束我們的ROS2程序,要釋放資源,比如要關閉context,這時已經關掉了context,這樣,我們再從client和this來獲取日誌,就不行了,所以建議用rclcpp::get_logger()。

因為rclcpp::get_logger()的調用和context沒有關係。

這樣程序就正常了!

#include "rclcpp/rclcpp.hpp"
#include "base_interfaces_demo/srv/add_ints.hpp"

using base_interfaces_demo::srv::AddInts;
using namespace std::chrono_literals;

class AddIntsClient: public rclcpp::Node
{
  public:
    AddIntsClient():Node("add_ints_client_node_cpp")
    {
      RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"客户端节点创建!");
      client_ = this->create_client<AddInts>("add_ints");
    }

    bool connect_server()
    {
      while(client_->wait_for_service(1s) != true)
      {
        RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"服务连接中!");

        if (rclcpp::ok() != true)
        {
          RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"强行终止客户端!");
          return false;
        }
      }
      return true;
    }

  private:
    rclcpp::Client<AddInts>::SharedPtr client_;
};

int main(int argc, char ** argv)
{
  if(argc != 3)
  {
    RCLCPP_ERROR(rclcpp::get_logger("rclcpp"),"请提交两个整形数字!");
    return 1;
  }
  rclcpp::init(argc,argv);

  auto client = std::make_shared<AddIntsClient>();

  bool flag = client->connect_server();

  if (flag != true)
  {
    RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"服务器连接失败,程序退出!");
    return 0;
  }

  rclcpp::shutdown();
  return 0;
}


返回值類型有了,我們就粘貼過去,

因為using base_interfaces_demo::srv::AddInts所以可以省略成AddInts

在主函數里要調用函數。

atoi()是把數據轉化成整形

我們還要處理響應,響應有3個

第一個是中斷,第二個是成功,第三個是超時;

我們一般只判斷成功,其他兩種情況都認為是失敗。

第一個入口參數是 節點的智能指針

第二個入口參數是future

第三個入口參數有默認值,先不用管

#include "rclcpp/rclcpp.hpp"
#include "base_interfaces_demo/srv/add_ints.hpp"

using base_interfaces_demo::srv::AddInts;
using namespace std::chrono_literals;

class AddIntsClient: public rclcpp::Node
{
  public:
    AddIntsClient():Node("add_ints_client_node_cpp")
    {
      RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"客户端节点创建!");
      client_ = this->create_client<AddInts>("add_ints");
    }

    bool connect_server()
    {
      while(client_->wait_for_service(1s) != true)
      {
        RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"服务连接中!");

        if (rclcpp::ok() != true)
        {
          RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"强行终止客户端!");
          return false;
        }
      }
      return true;
    }

    rclcpp::Client<AddInts>::FutureAndRequestId send_request(int32_t num1,int32_t num2)
    {
      /*
        返回值 rclcpp::Client<base_interfaces_demo::srv::AddInts>::FutureAndRequestId
        入口参数 async_send_request(std::shared_ptr<base_interfaces_demo::srv::AddInts_Request> request)  //其实就相当于AddInts::Request类型
      */
      auto request = std::make_sharedautolinkAddInts::Requestautolink();
      request->num1 = num1;
      request->num2 = num2;
      return client_->async_send_request(request);
    }

  private:
    rclcpp::Client<AddInts>::SharedPtr client_;
};

int main(int argc, char ** argv)
{
  if(argc != 3)
  {
    RCLCPP_ERROR(rclcpp::get_logger("rclcpp"),"请提交两个整形数字!");
    return 1;
  }
  rclcpp::init(argc,argv);

  auto client = std::make_shared<AddIntsClient>();

  bool flag = client->connect_server();

  if (flag != true)
  {
    RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"服务器连接失败,程序退出!");
    return 0;
  }
  auto future = client->send_request(atoi(argv[1]),atoi(argv[2]));

  if (rclcpp::spin_until_future_complete(client,future) == rclcpp::FutureReturnCode::SUCCESS)
  {
    RCLCPP_INFO(client->get_logger(),"响应成功! sum = %d",future.get()->sum);
  }
  else
  {
    RCLCPP_INFO(client->get_logger(),"响应失败!");
  }

  rclcpp::shutdown();
  return 0;
}

動作通信_理論

讓B一直給A返回當前機器人的狀態信息,這樣的行為通信更符合我們操控機器人的導航需求。

輸入10,會累加1-10的所有數,並且會遍歷1-10所有的數,並進行累加,累加是需要耗時的,假設每累加一次,耗時一秒,

然後為了好看出來程序運行情況,在每累加的時候,都發一個INFO,代表當前進度。

可以在進行任務時,把任務取消掉。

第一步,客戶端給服務端發目標數據

第二步,服務端評估目標數據,並反饋給客戶端這個評估結果(是否能夠達到目標)

第三步,客戶端再給服務端發最終確定的目標數據

第四步,服務端一直反饋給客戶端執行的過程數據

第五步,結束之後,服務端反饋給客戶端最終的結果


ros2 pkg create cpp03_action --build-type ament_cmake --dependencies rclcpp rclcpp_action base_interfaces_demo --node-name demo01_action_server


最頂上是請求數據,

中間是最終響應結果的數據,

最底下是連續反饋的數據。

depend是build depend,exe depend,export depend三者的集成。

ros2 interface show base_interfaces_demo/action/Progress

動作通信_實驗1_服務端實現(C++)

#include "rclcpp/rclcpp.hpp"

class ProgressActionServer: public rclcpp::Node
{
  public:
    ProgressActionServer():Node("progress_action_server_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"action服务端创建!");
    }
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  rclcpp::spin(std::make_shared<ProgressActionServer>());

  rclcpp::shutdown();
  return 0;
}

#include "rclcpp/rclcpp.hpp"

class ProgressActionClient: public rclcpp::Node
{
  public:
    ProgressActionClient():Node("progress_action_server_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"action客户端创建!");
    }
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  rclcpp::spin(std::make_shared<ProgressActionClient>());

  rclcpp::shutdown();
  return 0;
}


有倆模板,我們只需要設置action就行了,就是我們的動作接口類型。

第一個參數是node,在class裡就用this指針,

第二個參數是話題,字符串,

第三個參數是回調函數用來處理目標值的,

第四個參數是回調函數用來處理取消請求的,

第五個參數是接收目標值之後,該回調函數生成連續反饋,

第六、第七個參數有默認值,先不管,

返回值是action智能指針。

Goal_callback解析:

第一個參數是GoalUUID,

第二個參數是我們動作接口下的Goal,

返回值是goalresponse,用的命名空間是rclcpp_action,底下封裝了3個常量,

第一個是接收並馬上執行,

第二個是接收並推遲執行,

第三個是拒絕。

報錯原因是沒加佔位符

#include "rclcpp/rclcpp.hpp"
#include "rclcpp_action/rclcpp_action.hpp"
#include "base_interfaces_demo/action/progress.hpp"

using base_interfaces_demo::action::Progress;
using std::placeholders::_1;
using std::placeholders::_2;

class ProgressActionServer: public rclcpp::Node
{
  public:
    ProgressActionServer():Node("progress_action_server_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"action服务端创建!");
      /*
      rclcpp_action::Server<ActionT>::SharedPtr create_server<ActionT,
      NodeT>(NodeT node,
      const std::string &name,
      rclcpp_action::Server<ActionT>::GoalCallback handle_goal,
      rclcpp_action::Server<ActionT>::CancelCallback handle_cancel,
      rclcpp_action::Server<ActionT>::AcceptedCallback handle_accepted,
      const rcl_action_server_options_t &options = rcl_action_server_get_default_options(),
      rclcpp::CallbackGroup::SharedPtr group = nullptr)
      */
      server_ = rclcpp_action::create_server<Progress>(
        this,
        "get_sum_topic",
        std::bind(&ProgressActionServer::handle_goal_callback,this,_1,_2),
        std::bind(&ProgressActionServer::handle_cancel_callback,this,_1),
        std::bind(&ProgressActionServer::handle_accepted_callback,this,_1)
        );
    }
    //std::function<GoalResponse(const GoalUUID &, std::shared_ptr<const typename ActionT::Goal>)>
    rclcpp_action::GoalResponse handle_goal_callback(const rclcpp_action::GoalUUID &, std::shared_ptr<const Progress::Goal>)
    {

      return rclcpp_action::GoalResponse::ACCEPT_AND_EXECUTE;
    }

    //std::function<CancelResponse(std::shared_ptr<ServerGoalHandle<ActionT>>)>
    rclcpp_action::CancelResponse handle_cancel_callback(std::shared_ptr<rclcpp_action::ServerGoalHandle<Progress>> goal_handle)
    {

      return rclcpp_action::CancelResponse::ACCEPT;
    }

    //std::function<void (std::shared_ptr<ServerGoalHandle<ActionT>>)>
    void handle_accepted_callback(std::shared_ptr<rclcpp_action::ServerGoalHandle<Progress>> goal_handle)
    {
    }

  private:
    rclcpp_action::Server<Progress>::SharedPtr server_;
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  rclcpp::spin(std::make_shared<ProgressActionServer>());

  rclcpp::shutdown();
  return 0;
}

ros2 action send_goal /get_sum_topic base_interfaces_demo/action/Progress -f "{'num': 10}"

ros2 action send_goal /話題名稱 + 接口類型 + -f + 參數

-f是連續反饋,就是可以獲取連續反饋。

發送目標值為10

然後為我們客戶端設置了一個ID,因為可能有多個客戶端都訪問這個服務端,所以我們要給每個客戶端都設置一個唯一的ID

然後結果是0

因為我們程序暫時啥都還沒寫。


uuid就是客戶端ID,在此時沒有使用,那就用(void)uuid,因為如果不用,編譯器可能報警告。


我們的這個任務是可以正常被取消的,所以直接return accept就可以,根據不同任務需求來在函數里做不同的事。


因為我們的連續反饋和最終響應的生成是耗時操作,為了避免主邏輯出現阻塞,建議單獨再開一個線程。

生成連續反饋的API,需要傳參,傳入的參數就是Progress裡的Feedback對象。

get_goal()可以獲取目標值

因為這是個耗時操作,為了看出效果來,所以咱們每次循環的時候都給設置一下休眠。

1.0是指1Hz,也就是每隔1秒執行一次。

因為1Hz,所以我們這個rate.sleep()每次都會休眠1秒鐘;

生成最終結果的API,需要傳參,傳入的參數就是Progress裡的Result對象。

ok()函數的返回值:如果正常運行,則返回true,如果不正常運行則返回false


bug:當我們終止客戶端之後,服務端沒有停止運行。服務端要一直執行到程序結束。

bug解決思路:

接收到取消請求後,就是中斷我的主邏輯,也就是execute_callback被關閉,

這個函數返回值是布爾值,如果接收到了取消請求就返回true,否則返回false,

我們取消之後,其實仍然可以向客戶端反應最終的結果,用canceled函數,

要把定義result放在前面。

#include "rclcpp/rclcpp.hpp"
#include "rclcpp_action/rclcpp_action.hpp"
#include "base_interfaces_demo/action/progress.hpp"

using base_interfaces_demo::action::Progress;
using std::placeholders::_1;
using std::placeholders::_2;

class ProgressActionServer: public rclcpp::Node
{
  public:
    ProgressActionServer():Node("progress_action_server_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"action服务端创建!");
      /*
      rclcpp_action::Server<ActionT>::SharedPtr create_server<ActionT,
      NodeT>(NodeT node,
      const std::string &name,
      rclcpp_action::Server<ActionT>::GoalCallback handle_goal,
      rclcpp_action::Server<ActionT>::CancelCallback handle_cancel,
      rclcpp_action::Server<ActionT>::AcceptedCallback handle_accepted,
      const rcl_action_server_options_t &options = rcl_action_server_get_default_options(),
      rclcpp::CallbackGroup::SharedPtr group = nullptr)
      */
      server_ = rclcpp_action::create_server<Progress>(
        this,
        "get_sum_topic",
        std::bind(&ProgressActionServer::handle_goal_callback,this,_1,_2),
        std::bind(&ProgressActionServer::handle_cancel_callback,this,_1),
        std::bind(&ProgressActionServer::handle_accepted_callback,this,_1)
        );
    }

    //std::function<GoalResponse(const GoalUUID &, std::shared_ptr<const typename ActionT::Goal>)>
    rclcpp_action::GoalResponse handle_goal_callback(const rclcpp_action::GoalUUID &uuid, std::shared_ptr<const Progress::Goal> goal)
    {
      (void)uuid;
      if(goal->num <= 1)
      {
        RCLCPP_INFO(this->get_logger(),"提交的目标值必须大于1!");
        return rclcpp_action::GoalResponse::REJECT;
      }
      RCLCPP_INFO(this->get_logger(),"提交的目标值合法!");
      return rclcpp_action::GoalResponse::ACCEPT_AND_EXECUTE;
    }

    //std::function<CancelResponse(std::shared_ptr<ServerGoalHandle<ActionT>>)>
    rclcpp_action::CancelResponse handle_cancel_callback(std::shared_ptr<rclcpp_action::ServerGoalHandle<Progress>> goal_handle)
    {
      (void)goal_handle;
      RCLCPP_INFO(this->get_logger(),"接收到任务取消请求!");
      return rclcpp_action::CancelResponse::ACCEPT;
    }

    //std::function<void (std::shared_ptr<ServerGoalHandle<ActionT>>)>
    void execute_callback(std::shared_ptr<rclcpp_action::ServerGoalHandle<Progress>> goal_handle)
    {
      //void publish_feedback(std::shared_ptr<base_interfaces_demo::action::Progress_Feedback> feedback_msg)
      //goal_handle->publish_feedback();
      int num = goal_handle->get_goal()->num;
      int sum = 0;
      auto feedback = std::make_sharedautolinkProgress::Feedbackautolink();
      auto result = std::make_sharedautolinkProgress::Resultautolink();
      rclcpp::Rate rate(1.0);
      for (int32_t i = 1; i <= num; i++)
      {
        sum += i;
        double progress = i / (double)num;
        feedback->progress = progress;

        goal_handle->publish_feedback(feedback);
        RCLCPP_INFO(this->get_logger(),"连续反馈中,当前进度为:%.2f",progress);

        if(goal_handle->is_canceling() == true)
        {
          result->sum = sum;
          goal_handle->canceled(result);
          RCLCPP_INFO(this->get_logger(),"任务被取消了!");
          return;
        }

        rate.sleep();
      }

      //void succeed(std::shared_ptr<base_interfaces_demo::action::Progress_Result> result_msg)
      //goal_handle->succeed();

      if(rclcpp::ok() == true)
      {
        result->sum = sum;
        goal_handle->succeed(result);
        RCLCPP_INFO(this->get_logger(),"最终响应结果为:%d",sum);
      }

    }

    void handle_accepted_callback(std::shared_ptr<rclcpp_action::ServerGoalHandle<Progress>> goal_handle)
    {
      std::thread(std::bind(&ProgressActionServer::execute_callback,this,goal_handle)).detach();
    }

  private:
    rclcpp_action::Server<Progress>::SharedPtr server_;
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  rclcpp::spin(std::make_shared<ProgressActionServer>());

  rclcpp::shutdown();
  return 0;
}

動作通信_實驗1_客戶端實現(C++)

這條紅色的線是在action_client幫我們封裝好的,所以可以不用管。

#include "rclcpp/rclcpp.hpp"

class ProgressActionClient: public rclcpp::Node
{
  public:
    ProgressActionClient():Node("progress_action_server_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"action客户端创建!");
    }
};

int main(int argc, char ** argv)
{
  if(argc != 2)
  {
    RCLCPP_INFO(rclcpp::get_logger("rclcppp"),"请输入一个整形数字!");
    return 1;
  }
  rclcpp::init(argc,argv);

  rclcpp::spin(std::make_shared<ProgressActionClient>());

  rclcpp::shutdown();
  return 0;
}

第一個入口參數是node

第二個入口參數是話題名稱,字符串

第三個入口參數和第四個入口參數都有默認值

返回值是客戶端智能指針。

需要把一步分成兩步。

再把send_goal函數調用一下。


async_send_goal是異步發送目標值

第一個入口參數是我們接口文件裡的目標值

第二個入口參數是發送目標選項對象,我們可以設置這個目標發送過去之後,我們需要處理的一些回調函數

返回值是

先不管其他的,先把函數隨便定義上,什麼返回值,入口參數都是void,先不讓程序報錯。

得知,

GoalResponseCallback返回值是void,入口參數是goalhandle,goalhandle在本圖的上面,是clientgoalhandle,然後這個clientgoalhandle屬於rclcpp_action工作空間

返回值是void

第一個入口參數是clientgoalhandle

第二個入口參數是feedback


發送一個數值給服務端,服務端要先拿到目標值進行判斷,判斷該目標值是否可以被接收,或者被拒絕,再把處理結果響應給客戶端。

如果說這個目標值可處理,那麼goal_handle裡是有內容的;

如果不可以被處理,那麼goal_handle是一個nullptr空指針。


接下來處理反饋數據:

如果我們只是解析反饋的數據,那麼goal_handle是用不上的。

需要用倆%來轉譯%,如上圖是打印百分比數據的案例。

假設progress_int是50,則會輸出50%。

如果只想打印一個%,那就需要%%來轉譯。

回調函數是可能會數據丟失的。這是正常現象。


最終響應:

這個result最終結果的狀態是不一定的,有可能任務被取消了,或被終止了,也有可能任務正常運行了。

所以我們要通過狀態碼來判斷狀態。

第一個是被強行終止

第二個是取消

第三個是成功

第四個是未知

#include "rclcpp/rclcpp.hpp"
#include "rclcpp_action/rclcpp_action.hpp"
#include "base_interfaces_demo/action/progress.hpp"

using base_interfaces_demo::action::Progress;
using namespace std::chrono_literals;
using std::placeholders::_1;
using std::placeholders::_2;

class ProgressActionClient: public rclcpp::Node
{
  public:
    ProgressActionClient():Node("progress_action_server_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"action客户端创建!");
      /*
      rclcpp_action::Client<ActionT>::SharedPtr create_client<ActionT,
      NodeT>(NodeT node, const std::string &name,
      rclcpp::CallbackGroup::SharedPtr group = nullptr,
      const rcl_action_client_options_t &options = rcl_action_client_get_default_options())
      */
      client_ = rclcpp_action::create_client<Progress>(this,"get_sum_topic");
    }

    void send_goal(int32_t num)
    {
      if(client_->wait_for_action_server(1s) != true)
      {
        RCLCPP_ERROR(this->get_logger(),"服务连接失败!");
        return;
      }

      /*
      std::shared_future<rclcpp_action::ClientGoalHandle<base_interfaces_demo::action::Progress>::SharedPtr>
      async_send_goal(const base_interfaces_demo::action::Progress::Goal &goal,
      const rclcpp_action::Client<base_interfaces_demo::action::Progress>::SendGoalOptions &options)
      */
      auto goal = Progress::Goal();
      goal.num = num;

      rclcpp_action::Client<Progress>::SendGoalOptions options;
      options.goal_response_callback = std::bind(&ProgressActionClient::goal_response_callback,this,_1);
      options.feedback_callback = std::bind(&ProgressActionClient::feedback_callback,this,_1,_2);
      options.result_callback = std::bind(&ProgressActionClient::result_callback,this,_1);

      auto future = client_->async_send_goal(goal,options);
    }

  void goal_response_callback(rclcpp_action::ClientGoalHandle<Progress>::SharedPtr goal_handle)
  {
    if(goal_handle == nullptr)
    {
      RCLCPP_INFO(this->get_logger(),"目标请求被服务端拒绝!");
    }
    else
    {
      RCLCPP_INFO(this->get_logger(),"目标处理中!");
    }
  }

  void feedback_callback(rclcpp_action::ClientGoalHandle<Progress>::SharedPtr goal_handle,std::shared_ptr<const Progress::Feedback> feedback)
  {
    (void)goal_handle;
    double progress = feedback->progress;
    int progress_int = (int) (progress * 100);
    RCLCPP_INFO(this->get_logger(),"当前进度为:%d%%",progress_int);
  }

  void result_callback(const rclcpp_action::ClientGoalHandle<Progress>::WrappedResult & result)
  {
    if (result.code == rclcpp_action::ResultCode::SUCCEEDED)
    {
      RCLCPP_INFO(this->get_logger(),"最终结果为:%d",result.result->sum);
    }
    else if(result.code == rclcpp_action::ResultCode::ABORTED)
    {
      RCLCPP_INFO(this->get_logger(),"过程被中断!");
    }
    else if(result.code == rclcpp_action::ResultCode::CANCELED)
    {
      RCLCPP_INFO(this->get_logger(),"任务被取消!");
    }
    else
    {
      RCLCPP_INFO(this->get_logger(),"未知异常!");
    }
  }

  private:
    rclcpp_action::Client<Progress>::SharedPtr client_;
};

int main(int argc, char ** argv)
{
  if(argc != 2)
  {
    RCLCPP_ERROR(rclcpp::get_logger("rclcppp"),"请输入一个整形数字!");
    return 1;
  }
  rclcpp::init(argc,argv);

  auto node = std::make_shared<ProgressActionClient>();

  node->send_goal(atoi(argv[1]));

  rclcpp::spin(node);

  rclcpp::shutdown();
  return 0;
}

此時我們取消客戶端,反而服務端還在運行,這裡的ctrl+c只是結束了我們的客戶端,而不是指揮我們的客戶端去下發取消請求指令,我們只有去捕獲一下我們的鍵盤才能完成取消請求指令的發送。


修復bug:

還沒修復好

// 发送取消请求auto future_cancel = client_->async_cancel_goal(goal_handle);
rclcpp::spin_until_future_complete(this->get_node_base_interface(), future_cancel);
if (future_cancel.wait_for(1s) == std::future_status::ready)
{
  RCLCPP_INFO(this->get_logger(), "终止请求已发送!");
}
else
{
  RCLCPP_ERROR(this->get_logger(), "无法发送终止请求...");
}

參數服務_理論與API介紹(C++)

當然還有特殊情況,比如把參數設置為私有的。


其他通信需要自己弄接口文件,但是參數服務不用自己弄接口文件,ROS2已經封裝好了API,所以我們只需要調用API即可。

只是想展示一下API,所以先創建參數功能包展示下API,先不創建客戶端和服務端。

rclcpp::parameter 對象(鍵,值);

該函數是給parameter的值賦值的,有18個重載,各種類型。

其中空是說不給賦值,這樣只有鍵,沒有值。

參數服務_實驗1_服務端(C++)

這裡有差異,在Node裡傳入了第二個參數,這句是專門用來允許刪除參數的。undeclared解除聲明。

只有查詢和修改的接口API,並沒有新增和刪除的API,這是ROS2根據安全考慮的。

#include "rclcpp/rclcpp.hpp"

class ParamServer: public rclcpp::Node
{
  public:
    ParamServer():Node("param_server_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"参数服务端搭建!");
    }
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  auto node = std::make_shared<ParamServer>();

  rclcpp::spin(node);

  rclcpp::shutdown();
  return 0;
}
#include "rclcpp/rclcpp.hpp"

class ParamClient: public rclcpp::Node
{
  public:
    ParamClient():Node("param_client_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"参数客户端搭建!");
    }
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  auto node = std::make_shared<ParamClient>();

  rclcpp::spin(node);

  rclcpp::shutdown();
  return 0;
}


鏈式編程。

一個普通的節點就可以當參數服務端,不需要另行創建參數服務端。

#include "rclcpp/rclcpp.hpp"

class ParamServer: public rclcpp::Node
{
  public:
    ParamServer():Node("param_server_node_cpp",rclcpp::NodeOptions().allow_undeclared_parameters(true))
    {
      RCLCPP_INFO(this->get_logger(),"参数服务端搭建!");
    }

    void create_param()
    {
      RCLCPP_INFO(this->get_logger(),"-------------增操作--------------");
    }

    void get_param()
    {
      RCLCPP_INFO(this->get_logger(),"-------------查操作--------------");
    }

    void update_param()
    {
      RCLCPP_INFO(this->get_logger(),"-------------改操作--------------");
    }

    void delete_param()
    {
      RCLCPP_INFO(this->get_logger(),"-------------删操作--------------");
    }

};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  auto node = std::make_shared<ParamServer>();

  node->create_param();
  node->get_param();
  node->update_param();
  node->delete_param();

  rclcpp::spin(node);

  rclcpp::shutdown();
  return 0;
}

ros2 param list

查詢所有節點裡的所有參數

ros2 param get /節點名稱 參數鍵名 來查看參數的值


可以通過鍵來查詢參數的值

帶複數形式的函數可以通過由鍵組成的容器來獲取一些參數對象。

來判斷是否有該參數的,入口參數也是鍵,返回值是布爾值。


需要傳入parameter對象。

我們覆蓋掉舊值即可。

通過set_parameter也可以創建參數,但是必須聲明allow_undeclared_parameters(true)。


這種聲明的參數不可以被刪除,只能刪除未聲明但設置的。

#include "rclcpp/rclcpp.hpp"

class ParamServer: public rclcpp::Node
{
  public:
    ParamServer():Node("param_server_node_cpp",rclcpp::NodeOptions().allow_undeclared_parameters(true))
    {
      RCLCPP_INFO(this->get_logger(),"参数服务端搭建!");
    }

    void create_param()
    {
      RCLCPP_INFO(this->get_logger(),"-------------增操作--------------");

      this->declare_parameter("car_name","ER");
      this->declare_parameter("width",1.55);
      this->declare_parameter("wheels",5);

      this->set_parameter(rclcpp::Parameter("height",2.00));
    }

    void get_param()
    {
      RCLCPP_INFO(this->get_logger(),"-------------查操作--------------");

      auto car = this->get_parameter("car_name");
      RCLCPP_INFO(this->get_logger(),"key = %s,value = %s",car.get_name().c_str(),car.as_string().c_str()); 

      auto params = this->get_parameters({"car_name","width","wheels"});
      for(auto &¶m : params)
      {
        RCLCPP_INFO(this->get_logger(),"key = %s,value = %s",param.get_name().c_str(),param.value_to_string().c_str());
      }
      bool car_name_flag = this->has_parameter("car_name");
      bool height_flag = this->has_parameter("height");
      RCLCPP_INFO(this->get_logger(),"是否包含car_name? 答案:%d",car_name_flag);
      RCLCPP_INFO(this->get_logger(),"是否包含height? 答案:%d",height_flag);
    }

    void update_param()
    {
      RCLCPP_INFO(this->get_logger(),"-------------改操作--------------");
      this->set_parameter(rclcpp::Parameter("width",1.85));
      RCLCPP_INFO(this->get_logger(),"width = %.2f",this->get_parameter("width").as_double());
    }

    void delete_param()
    {
      RCLCPP_INFO(this->get_logger(),"-------------删操作--------------");
    //   this->undeclare_parameter("car_name");
    //   RCLCPP_INFO(this->get_logger(),"是否包含car_name? 答案:%d",this->has_parameter("car_name"));
      this->undeclare_parameter("height");
      RCLCPP_INFO(this->get_logger(),"是否包含height? 答案:%d",this->has_parameter("height"));
    }

};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  auto node = std::make_shared<ParamServer>();

  node->create_param();
  node->get_param();
  node->update_param();
  node->delete_param();

  rclcpp::spin(node);

  rclcpp::shutdown();
  return 0;
}

參數服務_實驗1_客戶端(C++)

第一個入口參數是客戶端節點對象,

第二個入口參數是需要連接的服務端的節點名稱。

如果1秒鐘之內連接上了就返回true,如果超時1s沒連接上就返回false。

#include "rclcpp/rclcpp.hpp"

using namespace std::chrono_literals;

class ParamClient: public rclcpp::Node
{
  public:
    ParamClient():Node("param_client_node_cpp")
    {
      RCLCPP_INFO(this->get_logger(),"参数客户端搭建!");
      param_client_ = std::make_sharedautolinkrclcpp::SyncParametersClientautolink(this,"param_server_node_cpp");
    }

    bool connect_server()
    {
      while(param_client_->wait_for_service(1s) != true)
      {
        if(rclcpp::ok() != true)
        {
          return false;
        }
        RCLCPP_INFO(this->get_logger(),"服务连接中!");
      }
      return true;
    }

    void get_param()
    {
      RCLCPP_INFO(this->get_logger(),"-----------参数查询操作-------------");
    }

    void update_param()
    {
      RCLCPP_INFO(this->get_logger(),"-----------参数更新操作-------------");
    }

  private:
    rclcpp::SyncParametersClient::SharedPtr param_client_;
};

int main(int argc, char ** argv)
{
  rclcpp::init(argc,argv);

  auto client = std::make_shared<ParamClient>();

  bool flag = client->connect_server();

  if(!flag)
  {
    return 0;
  }

  client->get_param();
  client->update_param();
  client->get_param();

  // rclcpp::spin(client);

  rclcpp::shutdown();
  return 0;
}

這些話題都是我們此節點名稱下的。


要用高級for

入口參數填參數對象的容器。

我們不僅可以修改值,也可以創建新的參數,但是要保證服務端那邊調用過undeclared......

音乐页