ros2学习笔记-通信接口

目录

1.ros2接口描述

2.ros2接口定义

3.ROS2接口常用的命令行指令

4.测试自定义接口 


1.ros2接口描述

        ros2节点之间通讯一般使用std_msgs/下的标准数据类型,当我们要定义自己的消息类型的时候,需要定义通讯接口(数据类型)文件。自定义的通讯数据类型可以用已有的通讯数据类型进行组装,也可以使用ros2规定的原始数据类型组装。已有的通讯接口定义可以使用ros2的命令查看:

ros2 interface list

ros2规定的原始的数据类型只有九类。其中每一个都可以在后面加上[]将其变成数组形式:

bool
byte
char
float32, float64
int8, uint8
int16, uint16
int32, uint32
int64, uint64
string

 2.ros2接口定义

         在ROS2中根据不同通讯类型定义不同接口文件,该文件后缀分别为msgsrvaction.

例如:

话题接口

文件名: *.msg

int64 num

服务接口

文件名:*.srv

int64 a    #这三个是输入参数
int64 b
int64 c
---
int64 sum   #这个是返回结果

动作接口

文件名:*.action

int32 order
---
int32[] sequence
---
int32[] partial_sequence

至于参数这种方式不存在接口定义。 

以话题通讯方式举例定义接口:

  1. 首先使用 命令创建包(注意:这里使用的是c++编译方式)。
    1. ros2 pkg create test --dependencies rclcpp --build-type ament_cmake 
      
  2.  在CmakeLists.txt文件存在的目录建立msg目录 ,并在msg文件夹下创建消息接口文件,后缀名必须是 .msg ,文件名字可以随便取名字。
    1. mkdir msg
      touch name.msg
  3.  在name.msg文件里面添加消息内容并保存。
  4. 在CmakeLists.txt里面添加name.msg文件需要的依赖包和msg文件目录。
  5. 在package.xml中添加name.msg需要的依赖。
  6. 编译功能包即可生成python和c++需要的头文件。

如图所示:

1.添加消息内容(注意文件名:Name.msg首字母必须大写)

2.编辑CmakeLists.txt文件(注意添加位置)

# 这两句添加依赖
find_package(sensor_msgs REQUIRED)
find_package(rosidl_default_generators REQUIRED)

# 声明msg文件所属的工程名字, 文件位置, 依赖DEPENDENCIES
rosidl_generate_interfaces(${PROJECT_NAME}
  "msg/Name.msg"
   DEPENDENCIES sensor_msgs
 )

3.修改package.xml

<!-- 添加如下内容 -->
<depend>sensor_msgs</depend>
<build_depend>rosidl_default_generators</build_depend>
<exec_depend>rosidl_default_runtime</exec_depend>
<member_of_group>rosidl_interface_packages</member_of_group>

4.执行编译生成c++头文件

colcon build

会在build/interface路径下生成一系列目录文件 

3.ROS2接口常用的命令行指令

  • 查看接口列表ros2 interface list
  • 查看所有接口包ros2 interface packages
  • 查看某一个包下的所有接口ros2 interface package std_msgs
  • 查看某一个接口详细的内容 ros2 interface show std_msgs/msg/String
  • 输出某一个接口所有属性 ros2 interface proto sensor_msgs/msg/Image

4.测试自定义接口 

编写订阅发布程序

intertest.h文件

#ifndef INTERTEST_H
#define INTERTEST_H
#include "rclcpp/rclcpp.hpp"
#include "interface/msg/name.hpp"
#include "rclcpp/publisher.hpp"
#include "rclcpp/subscription.hpp"
#include "std_msgs/msg/string.h"
class intertest:public rclcpp::Node
{
public:
    intertest(std::string text);
//private:
    void interfaceCallback(interface::msg::Name msg);
    void timerCallback();
private:
//    interface::msg::Name
    rclcpp::Publisher<interface::msg::Name>::SharedPtr puber_;
    rclcpp::Subscription<interface::msg::Name>::SharedPtr Suber_;
    rclcpp::TimerBase::SharedPtr timerfun_;

//    rclcpp::TimerBase::SharedPtr timer_;
};

#endif // INTERTEST_H

 intertest.cpp文件

#include "intertest.h"

using ::std::placeholders::_1;

intertest::intertest(std::string text):
    rclcpp::Node(text)
{
    RCLCPP_INFO(this->get_logger(),"%s",text.c_str());
    puber_ = this->create_publisher<interface::msg::Name>("interface_test",10);
    Suber_ = this->create_subscription<interface::msg::Name>("interface_test",10,
                                                     std::bind(&intertest::interfaceCallback,this,_1));

   timerfun_ =  this->create_wall_timer(std::chrono::milliseconds(100),std::bind(&intertest::timerCallback,this));
}

void intertest::interfaceCallback(interface::msg::Name msg)
{
    RCLCPP_INFO(this->get_logger(),"%s,%d",msg.text.c_str(),msg.count);
}

void intertest::timerCallback()
{
    static interface::msg::Name msg;
    msg.count++;
    msg.text = "interfacetest";
    puber_->publish(msg);
}

main.cpp文件

#include "intertest.h"

int main(int argc,char** argv)
{
    rclcpp::init(argc,argv);
    auto p = std::make_shared<intertest>("test");
    rclcpp::spin(p);
    rclcpp::shutdown();
    return 0;
}

修改CMakeLists.txt

find_package(interface REQUIRED)

add_executable(inter src/main.cpp src/intertest.cpp)
ament_target_dependencies(inter "rclcpp" "std_msgs" "interface")

install(TARGETS
  inter
  DESTINATION lib/${PROJECT_NAME}
)
 

make_minimum_required(VERSION 3.8)
project(interface)

if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang")
  add_compile_options(-Wall -Wextra -Wpedantic)
endif()

# find dependencies
find_package(ament_cmake REQUIRED)
find_package(rclcpp REQUIRED)
find_package(std_msgs REQUIRED)
# 这三句添加依赖
find_package(sensor_msgs REQUIRED)
find_package(rosidl_default_generators REQUIRED)
find_package(interface REQUIRED)
# 声明msg文件所属的工程名字, 文件位置, 依赖DEPENDENCIES
rosidl_generate_interfaces(${PROJECT_NAME}
  "msg/Name.msg"
   DEPENDENCIES sensor_msgs
 )

if(BUILD_TESTING)
  find_package(ament_lint_auto REQUIRED)
  # the following line skips the linter which checks for copyrights
  # comment the line when a copyright and license is added to all source files
  set(ament_cmake_copyright_FOUND TRUE)
  # the following line skips cpplint (only works in a git repo)
  # comment the line when this package is in a git repo and when
  # a copyright and license is added to all source files
  set(ament_cmake_cpplint_FOUND TRUE)
  ament_lint_auto_find_test_dependencies()
endif()

ament_package()




 # target_compile_features(interface PUBLIC c_std_99 cxx_std_17)  # Require C99 and C++17
add_executable(inter src/main.cpp src/intertest.cpp)
ament_target_dependencies(inter "rclcpp" "std_msgs" "interface")

install(TARGETS
  inter
  DESTINATION lib/${PROJECT_NAME}
)

package.xml保持原样不用修改

<?xml version="1.0"?>
<?xml-model href="http://download.ros.org/schema/package_format3.xsd" schematypens="http://www.w3.org/2001/XMLSchema"?>
<package format="3">
  <name>interface</name>
  <version>0.0.0</version>
  <description>TODO: Package description</description>
  <maintainer email="klppc@todo.todo">klppc</maintainer>
  <license>TODO: License declaration</license>

  <buildtool_depend>ament_cmake</buildtool_depend>

  <depend>rclcpp</depend>
  <!-- 添加如下内容 -->
  <depend>sensor_msgs</depend>
  <depend>std_msgs</depend>
  <build_depend>rosidl_default_generators</build_depend>
  <exec_depend>rosidl_default_runtime</exec_depend>
  <member_of_group>rosidl_interface_packages</member_of_group>

  <test_depend>ament_lint_auto</test_depend>
  <test_depend>ament_lint_common</test_depend>

  <export>
    <build_type>ament_cmake</build_type>
  </export>
</package>

编译程序:

source ./install/setup.bash
colcon build
ros2 run interface inter

运行结果:

rqt查看:

注意:使用命令打印topic消息时需要先source一下环境。

source ./cache/interface/install/setup.bash
ros2 topic echo /interface_test

参考资料:

ROS2——什么是接口_ros2 std_msgs_范子琦的博客-CSDN博客

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值