跳至主要內容

进行 ROS 实验

Alex Sun2023年3月21日环境搭建ROS大约 4 分钟

ROS 实验内容。

说明

本文的源代码在 Gitee 上保存,在进行本文实验时需要确保安装了 ROS 环境安装 一文中的指定环境。

1. 自定义话题通信

创建一个名为 learning01_ws 的工作空间:

mkdir -p learning01_ws/src
cd learning01_ws/
catkin_make

创建名为 plumbing_pub_sub 的功能包并添加相关依赖:

  1. 在 VS Code 中按 Ctrl + Shift + P 使用命令面板,输入 ROS: Create Catkin Package 回车
  2. 接着输入包名 plumbing_pub_sub
  3. 输入依赖项 roscpp rospy std_msgs

新建文件 plumbing_pub_sub/msg/Person.msg,内容是:

string  location
int32   number
float32 floor

编辑 plumbing_pub_sub/package.xml,在 <exec_depend> 标签附近添加下面两句:

  <build_depend>message_generation</build_depend>
  <exec_depend>message_runtime</exec_depend>

编辑 plumbing_pub_sub/CMakeLists.txt

# 需要替换
find_package(catkin REQUIRED COMPONENTS
  roscpp
  rospy
  std_msgs
  message_generation
)

# 直接添加
add_message_files(
  FILES
  Person.msg
)

# 直接添加
generate_messages(
  DEPENDENCIES
  std_msgs
)

# 需要替换
catkin_package(
  CATKIN_DEPENDS roscpp rospy std_msgs message_runtime
)

现在执行构建,按 Ctrl + Shift + B,然后选中 catkin_make: build,然后回车。如果配置正确将完成构建。

在编写 Python 代码之前,最好在 .vscode/settings.json 中配置 Python 的读取路径:

{
    "python.autoComplete.extraPaths": [
        "/opt/ros/noetic/lib/python3/dist-packages",
        "${workspaceFolder}/devel/lib/python3/dist-packages"
    ],
    "python.analysis.extraPaths": [
        "/opt/ros/noetic/lib/python3/dist-packages",
        "${workspaceFolder}/devel/lib/python3/dist-packages"
    ]
}

plumbing_pub_sub/ 文件夹下创建 scripts/demo03_pub_person_p.pyscripts/demo04_sub_person_p.py,下面是创建命令,并赋予可执行权限:

cd src/plumbing_pub_sub/
mkdir -p scripts
cd script
touch demo03_pub_person_p.py
touch demo04_sub_person_p.py

chmod +x *.py

demo03_pub_person_p.py 文件的内容是:

#!/usr/bin/env python3

import rospy
from plumbing_pub_sub.msg import Person

if __name__ == '__main__':
    rospy.init_node('AlexPub')
    pub = rospy.Publisher('Alex', Person, queue_size=10)
    p = Person()
    p.location = 'ChangZhou'
    p.number = 66
    p.floor = 1.66
    rate = rospy.Rate(1)
    while not rospy.is_shutdown():
        pub.publish(p)
        rate.sleep()
        rospy.loginfo('House data: %s, %d, %.2f', p.location, p.number, p.floor)

demo04_sub_person_p.py 文件的内容是:

#!/usr/bin/env python3

import rospy
from plumbing_pub_sub.msg import Person

def do_person(person: Person):
    rospy.loginfo('House data: %s, %d, %.2f', person.location, person.number, person.floor)

if __name__ == '__main__':
    rospy.init_node('AlexSub')
    sub = rospy.Subscriber('Alex', Person, do_person, queue_size=10)
    rospy.spin()

现在,编辑 plumbing_pub_sub/CMakeLists.txt,在下方添加下面的代码:

catkin_install_python(PROGRAMS scripts/demo03_pub_person_p.py
  DESTINATION ${CATKIN_PACKAGE_BIN_DESTINATION}
)

catkin_install_python(PROGRAMS scripts/demo04_sub_person_p.py
  DESTINATION ${CATKIN_PACKAGE_BIN_DESTINATION}
)

然后按 Ctrl + Shift + B,再次执行编译。编译成功后可测试。

下面创建三个终端,执行测试:

终端 1
# 启动 ROS 核心进程,只需要启动一次
roscore

使用下面的命令用于可视化查看当前 ROS 通信的逻辑图:

rqt_graph

如果两个终端都分别看到消息表明成功。

2. 服务通信

创建功能包名为 plumbing_server_client 并添加相关依赖:

  1. 在 VS Code 中按 Ctrl + Shift + P 使用命令面板,输入 ROS: Create Catkin Package 回车
  2. 接着输入包名 plumbing_server_client
  3. 输入依赖项 roscpp rospy std_msgs

在功能包下创建 scripts 文件夹和 srv 文件夹,创建文件 srv/Multiply.srv,其内容为:

# 客户端请求时发送的两个数字
int32 num1
int32 num2
---
# 服务器响应发送的数据
int32 sum

编辑 plumbing_server_client/package.xml,在 <exec_depend> 标签附近添加下面两句:

  <build_depend>message_generation</build_depend>
  <exec_depend>message_runtime</exec_depend>

编辑 plumbing_server_client/CMakeLists.txt

# 需要替换
find_package(catkin REQUIRED COMPONENTS
  roscpp
  rospy
  std_msgs
  message_generation
)

# 直接添加
add_service_files(
  FILES
  Multiply.srv
)

# 直接添加
generate_messages(
  DEPENDENCIES
  std_msgs
)

# 需要替换
catkin_package(
  CATKIN_DEPENDS roscpp rospy std_msgs message_runtime
)

现在执行构建,步骤和上面一样。确保生成成功继续。

创建 scripts/demo01_server_p.py,内容为:

#!/usr/bin/env python3

import rospy

from plumbing_server_client.srv import (Multiply, MultiplyRequest,
                                        MultiplyResponse)

def do_req(req: MultiplyRequest):
    res = req.num1 * req.num2
    rospy.loginfo('got data: %d x %d = %d', req.num1, req.num2, res)
    resp = MultiplyResponse(res)
    resp.sum = res
    return resp

if __name__ == '__main__':
    rospy.init_node('MulService')
    server = rospy.Service('MultiPly', Multiply, do_req)
    rospy.spin()

创建 scripts/demo02_client_p.py,内容为:

#!/usr/bin/env python3

import sys

import rospy

from plumbing_server_client.srv import Multiply, MultiplyResponse

if __name__ == '__main__':
    if len(sys.argv) != 3:
        rospy.logerr('Params Error!')
        sys.exit(1)
    rospy.init_node('MulCli')
    client = rospy.ServiceProxy('MultiPly', Multiply)
    num1 = int(sys.argv[1])
    num2 = int(sys.argv[2])
    client.wait_for_service()
    res: MultiplyResponse = client.call(num1, num2)
    rospy.loginfo('got res: %d', res.sum)

加入下面的代码到 CMakeLists.txt

catkin_install_python(PROGRAMS scripts/demo01_server_p.py
  DESTINATION ${CATKIN_PACKAGE_BIN_DESTINATION}
)

catkin_install_python(PROGRAMS scripts/demo02_client_p.py
  DESTINATION ${CATKIN_PACKAGE_BIN_DESTINATION}
)

现在授予权限并运行服务,测试结果。

chmod +x plumbing_server_client/scripts/*.py

现在执行构建,步骤和上述构建步骤一致,确保构建成功。

下面创建三个终端,执行测试:

终端 1
# 启动 ROS 核心进程,只需要启动一次
roscore

终端 3 用于计算两个数的乘积,需要在命令行指定这两个数字。

3. 参数服务器

创建功能包名为 plumbing_param_server 并添加相关依赖。

创建文件 plumbing_param_server/scripts/demo01_param_set_p.py,内容为:

#!/usr/bin/env python3

import rospy

if __name__ == '__main__':
    rospy.init_node('param_set_p')
    rospy.set_param('name', 'alex')
    rospy.set_param('radius', 0.25)
    rospy.set_param('weight', 60)
    rospy.set_param('speed', 2.33)

创建 demo02_param_get_p.py 文件,内容为:

#!/usr/bin/env python3

import rospy

if __name__ == '__main__':
    rospy.init_node('param_get_p')
    key = rospy.search_param('radius')
    rospy.loginfo('key = %s', key)
    if key is not None:
        radius = rospy.get_param('radius')
        rospy.loginfo('radius = %.2f', radius)

创建 demo03_param_del_p.py,内容为:

#!/usr/bin/env python3

import rospy

if __name__ == '__main__':
    rospy.init_node('param_del_p')
    try:
        rospy.delete_param('radius')
    except:
        rospy.loginfo('Delete Error!')

添加可执行权限:

chmod +x src/plumbing_param_server/scripts/*.py

CMakeLists.txt 中加入下面的代码:

catkin_install_python(PROGRAMS scripts/demo01_param_set_p.py
  DESTINATION ${CATKIN_PACKAGE_BIN_DESTINATION}
)

catkin_install_python(PROGRAMS scripts/demo02_param_get_p.py
  DESTINATION ${CATKIN_PACKAGE_BIN_DESTINATION}
)

catkin_install_python(PROGRAMS scripts/demo03_param_del_p.py
  DESTINATION ${CATKIN_PACKAGE_BIN_DESTINATION}
)

现在构建,确保没有错误即可运行。

利用 rosparam list 命令查看所有参数,利用 rosparam get <param> 命令获取指定参数。

下面创建三个终端,执行测试:

终端 1
# 启动 ROS 核心进程,只需要启动一次
roscore