Python-PyQt-ROS通信

5 篇文章 0 订阅
1 篇文章 0 订阅

前言

建立与开源机器人系统ROS的通信,并基于QT创建应用程序,用于操作机器人。
由于ROS环境搭建过于繁琐,在这里仅仅是与ROS建立通信进行交互,因此,采用python的roslibpy库来进行通信连接。

Python ROS Bridge 库允许使用 Python 和 IronPython 与开源机器人中间件ROS进行交互。它使用 WebSockets 连接到 rosbridge 2.0并提供发布、订阅、服务调用、actionlib、TF 和其他基本 ROS 功能。
与rospy库不同,它不需要本地 ROS 环境,允许从 Linux 以外的平台使用。

roslibpy库详细内容见官网https://roslibpy.readthedocs.io/en/latest/
中文文档见https://roslibpy-docs-zh.readthedocs.io/zh/latest/
建议大家看官网,能够看到其具体实现。

一、安装并引入库

安装,命令行:

pip install roslibpy

引入库:

import roslibpy

二、使用方法

创建一个对象连接ROS

#连接ROS
self.ros = roslibpy.Ros(host=self.host, port=self.port)

运行

self.ros.run(timeout=2)

检测连接状态

if self.ros and self.ros.is_connected:
    self.ros.close()

断开连接

self.ros.close() #断开 websocket 连接. 在连接被关闭后,通过调用
self.ros.connect() #还可以再次连接。
self.ros.terminate() #终止主事件循环。如果连接还是打开着的,那么就会首先关闭它订阅一个消息

订阅话题

#订阅话题
self.get_data = roslibpy.Topic(self.ros, "/slamware_ros_sdk_server_node/odom", "nav_msgs/Odometry")
self.get_data.subscribe(callback=self.setRobotdata) #收到数据,执行回调

# 接收数据,并处理及执行其它动作
def setRobotdata(self, data):
#/*
#对接收到的数据data进行处理,或执行其它动作
#*/
    pass

发布话题

#发布话题
self.move_to = roslibpy.Topic(self.ros, "/slamware_ros_sdk_server_node/move_to",
                              "slamware_ros_sdk/MoveToRequest")
#发送数据
self.move_to.publish(roslibpy.Message(数据内容))

使用服务

#使用服务
self.get_last_map = roslibpy.Service(self.ros, "/slamware_ros_sdk_server_node/get_last_map",
                                      "slamware_ros_sdk/GetLastMap")
#发送请求并等待响应
result = self.get_last_map.call(roslibpy.ServiceRequest({请求内容}), timeout=10) 
#result为响应数据

活动客户端

#客户端使用ROS活动,
self.move_to_action = actionlib.ActionClient(self.ros, "/MoveActionServer", "slamware_ros_sdk/Move__Action")
goal = actionlib.Goal(self.move_to_action,roslibpy.Message(数据内容))
goal.send(result_callback=s_callback)  #发送成功,执行回调s_callback         

三、简单的ROS通信建立

部分代码:

import roslibpy, time
from enum import Enum
from PyQt5.QtCore import pyqtSignal, QThread
from roslibpy import actionlib
from roslibpy.actionlib import ActionClient

class RobotClient(QThread):
    move_to_action: ActionClient
    
    #创建信号
    map_ready = pyqtSignal(dict)
    robot_pos_ready = pyqtSignal(dict)
    scan_ready = pyqtSignal(dict)
    plan_path_ready = pyqtSignal(dict)
    robot_basic_state_ready = pyqtSignal(dict)
    virtual_walls_ready = pyqtSignal(dict)
    stoptask = pyqtSignal(bool)
    error_happened = pyqtSignal(str)

    connect_robot = pyqtSignal(bool)
    map_robot = pyqtSignal()
    map_loal = pyqtSignal()
    mgsdisplay = pyqtSignal()
    
    def __init__(self, parent=None):
        super(RobotClient, self).__init__(parent)
        self.host = None
        self.port = None
        self.ros = None

    def __del__(self):
        if self.ros and self.ros.is_connected:
            self.ros.close()
	# ==========================初始化ROS连接==========================
    def __initRos(self):
        try:
            #连接ROS
            self.ros = roslibpy.Ros(host=self.host, port=self.port)
            #订阅地图数据
            self.get_map = roslibpy.Topic(self.ros, "/slamware_ros_sdk_server_node/map", "nav_msgs/OccupancyGrid")
            self.get_map.subscribe(callback=self.setMap)
            #订阅位姿数据
            self.get_odom = roslibpy.Topic(self.ros, "/slamware_ros_sdk_server_node/odom", "nav_msgs/Odometry")
            self.get_odom.subscribe(callback=self.setRobotPos)
            #订阅激光数据
            self.get_scan = roslibpy.Topic(self.ros, "/slamware_ros_sdk_server_node/scan", "sensor_msgs/LaserScan")
            self.get_scan.subscribe(callback=self.setScan)
            #订阅路径数据
            self.get_plan_path = roslibpy.Topic(self.ros, "/slamware_ros_sdk_server_node/global_plan_path",
                                                "nav_msgs/Path")
          	#请求获取最新的地图
            self.get_last_map = roslibpy.Service(self.ros, "/slamware_ros_sdk_server_node/get_last_map",
                                                 "slamware_ros_sdk/GetLastMap")
         except Exception as e:
            print(e)
#被其他类引用
    def isConnect(self):
        if self.ros and self.ros.is_connected:
            return True
        else:
            return False  
            
   def setRos(self, host, port):
    try:
        if self.host != host or self.port != port:
            # 初始/切换
            if self.ros:
                # 切换的时候,结束上一次的连接,这里使用该库的on_ready方法的具体实现的部分代码
                def _wrapper_callback(proto):
                    proto.send_close()
                    return proto
                self.ros.factory.on_readyon_(_wrapper_callback)
            self.host = host
            self.port = port
            self.__initRos()
            return True
        else:
            if not self.ros.is_connected:
                # 重连
                return True
            else:
                return False
    except Exception as e:
        print(e)
        return False 
   # ==========================开启ROS,被其他类引用==========================
    def start_run(self):
        try:
            self.ros.run(timeout=2)
            #客户端使用ROS活动,
            self.move_to_action = actionlib.ActionClient(self.ros, "/MoveActionServer", "slamware_ros_sdk/Move__Action")
            return True
        except Exception as e:
            print(e)
            return False 
   
   #=========Robot线程=======================================================
    def run(self):
        # --------------------------点击了连接机器人--------------------------
        # while True:
        if self.parent().connectrobot:
            zRobot = self.parent().connectrobot
            if self.setRos(host=zRobot["Host"], port=zRobot["Port"]):
                self.succend = self.start_run()
                self.parent().connectrobot = None
                self.connect_robot.emit(self.succend)
                   
   # 位姿数据
    def setRobotPos(self, msg):
        # 位姿数据
        self.msg = msg
        self.robot_pos_ready.emit(msg)
        # 激光数据
    def setScan(self, msg):
        # 激光数据
        self.scan_ready.emit(msg)
    # 路径数据
    def setPlanPath(self, msg):
        # 路径数据
         self.plan_path_ready.emit(msg)
	# 获取最新的地图
    def doGetLastMap(self):
        # 获取最新的地图
        try:
            result = self.get_last_map.call(roslibpy.ServiceRequest({}), timeout=10)
            return result
        except Exception as e:
            print(e)
            return False
  	#ROS活动
	def doMoveToAction(self, point, s_callback, e_callback):
	#ROS活动
	 try:
	     goal = actionlib.Goal(self.move_to_action,
	                           roslibpy.Message(
	                               {"locations": [{"x": point.PositionX, "y": point.PositionY, "z": point.PositionZ}],
	                                "options": {"opt_flags": {"flags": 0x00000080},
	                                            "speed_ratio": {"is_valid": False, "value": 0}},
	                                "yaw": 0}))
	     goal.send(result_callback=s_callback)
	 except Exception as e:
	     print(e)                                       

roslibpy库on_ready的具体实现:

def on_ready(self, callback, run_in_thread=True):
        """Add a callback to be executed when the connection is established.

        If a connection to ROS is already available, the callback is executed immediately.

        Args:
            callback: Callable function to be invoked when ROS connection is ready.
            run_in_thread (:obj:`bool`): True to run the callback in a separate thread, False otherwise.
        """
        def _wrapper_callback(proto):
            if run_in_thread:
                self.factory.manager.call_in_thread(callback)
            else:
                callback()

            return proto

        self.factory.on_ready(_wrapper_callback)

总结

上面是我运用roslibpy库,建立的ROS通信。
如有错误希望请大家指导,谢谢点赞!
希望和大家一起学习,交流!

  • 5
    点赞
  • 44
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Python-pyqt5-hexview是一种基于PythonPyQt5这两个开发工具创建的十六进制数据查看器,是一个实用的工具,可以方便地查看二进制、十六进制数据。它提供了一种方便的方法来查看和编辑二进制文件,可以快速鉴定和分析数据,是开发人员、安全研究人员和网络管理员的理想选择。 Python-pyqt5-hexview可以明显提高开发人员的编程效率,节省时间和精力。它提供了一个友好的图形用户界面,用户可以在这个界面中选择要查看的数据文件,并自动将其转换成十六进制格式。具体而言,用户可以通过界面的文件浏览器来选择所需文件并打开文件。接下来,Python-pyqt5-hexview将自动将其读入并将数据转换成十六进制形式,并将其呈现在界面上。 Python-pyqt5-hexview支持多种常见的数据格式,以便用户可以轻松地查看和编辑复杂的数据文件。此外,它还具有许多其他功能,包括选择数据的起始和结束位置、搜索特定的十六进制模式、支持编辑和自动保存等。总之,Python-pyqt5-hexview是一个极为实用的数据查看器,为用户提供了一种方便、快速和高效的方式来查看和分析二进制和十六进制数据。 ### 回答2: Python-pyqt5-hexview是一种Python的库,是一个用于视觉化hex编辑器的插件。它使用PyQt5创建了一个用户界面,使用户能够以字节的形式查看和编辑二进制文件。此外,它还提供了搜索和替换功能,使用户能够快速准确地定位到需要编辑的信息。Python-pyqt5-hexview易于使用,具有良好的用户界面,是一款非常实用的工具。它可以被应用于各种领域,如数据恢复和修复、漏洞分析和漏洞利用。通过Python-pyqt5-hexview,用户可以查看和编辑数据,发现和利用漏洞,更好地理解和分析二进制文件。此外,它还可以被用于创建自定义hex编辑器,满足特定需求的个性化开发。总之,Python-pyqt5-hexview是一个功能强大的工具,为大量二进制文件的处理提供了方便、快捷的途径。 ### 回答3: Python-PyQt5-Hexview是一个PythonPyQt5编写的十六进制查看器,可以用于查看和编辑二进制文件。它具有用户友好的界面和丰富的操作功能,可以以字节为单位显示文件内容,并支持多种数据视图,方便用户查看和编辑不同格式的数据。此外,Python-PyQt5-Hexview还支持从文件和剪贴板中导入和导出数据,用户可以轻松地将文件内容复制到剪贴板并粘贴到其他应用程序中。除此之外,用户还可以将文件保存为常见的二进制文件格式,如.HEX或.BIN,方便用户进行后续的处理和分析。Python-PyQt5-Hexview还可以进行搜索和替换操作,用户可以快速定位和修改文件中的关键信息。Python-PyQt5-Hexview支持Windows、Linux和MacOS等多个操作系统,用户可以在不同平台上轻松使用它。总之,Python-PyQt5-Hexview是一个功能强大且易于使用的十六进制查看器,可以帮助用户快速查看和编辑二进制文件,是程序员和安全专家必备的工具之一。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值