基础通信
ROS2的基础通信
通信模式 | 作用 | 特点 | 应用场景 |
---|---|---|---|
话题:发布-订阅 (Publish-Subscribe) | 单向、异步数据流 | 高效、解耦,适用于传感器数据流、状态更新 | 传感器数据流(如相机图像、激光雷达数据)、机器人状态更新(如位置、速度) |
服务:服务端-客户端 (Service-Client) | 同步、请求-响应 | 阻塞式通信,用于需要明确结果的任务 | 触发特定动作(如抓取物体)、查询物体 ID |
动作 (Actions) | 长时任务处理 | 异步、支持进度反馈和取消 | 移动机器人到指定位置、机械臂执行复杂任务 |
参数 (Parameters) | 动态配置节点行为 | 非实时通信,用于启动时或运行时的配置调整 | 调整 PID 控制器增益、修改传感器发布频率 |
话 题
- 运行话题的发布者节点 :
ros2 run learning_topic topic_helloworld_pub
- 运行话题的订阅者节点 :
ros2 run learning_topic topic_helloworld_sub
可以看到发布者循环发布“Hello World”字符串消息,订阅者也以几乎同样的频率收到该话题的消息数据。
服务
- 运行服务的服务端节点 :
ros2 run learning_service service_adder_server
- 运行服务的客户端节点 :
ros2 run learning_service service_adder_client 2 3
动作
- 运行动作的服务端节点 :
ros2 run learning_action action_move_server
- 运行动作的客户端节点 :
ros2 run learning_action action_move_client
参数
等价于机器学习中的超参数
即:在开始训练前设置的参数,例如:学习率、隐藏层大小、迭代次数等等。
- 运行节点 :
ros2 run learning_parameter param_declare
- 在另外终端设置参数 :
ros2 param set param_declare robot_name turtle
练习
修改下面的代码。尝试传入不同的参数,将图片进行不同的处理。
调用同网段的摄像头
sender.py
import os
from importlib import import_module
from flask import Flask, render_template, Response,render_template_string
from io import BytesIO
import cv2
from PIL import ImageGrab, Image
import time
import threading
try:
from greenlet import getcurrent as get_ident
except ImportError:
try:
from thread import get_ident
except ImportError:
from _thread import get_ident
class CameraEvent(object):
def __init__(self):
self.events = {}
def wait(self):
ident = get_ident()
if ident not in self.events:
self.events[ident] = [threading.Event(), time.time()]
return self.events[ident][0].wait()
def set(self):
now = time.time()
remove = None
for ident, event in self.events.items():
if not event[0].isSet():
event[0].set()
event[1] = now
else:
if now - event[1] > 5:
remove = ident
if remove:
del self.events[remove]
def clear(self):
self.events[get_ident()][0].clear()
class BaseCamera(object):
thread = None
frame = None
last_access = 0
event = CameraEvent()
def __init__(self):
if BaseCamera.thread is None:
BaseCamera.last_access = time.time()
BaseCamera.thread = threading.Thread(target=self._thread)
BaseCamera.thread.start()
while self.get_frame() is None:
time.sleep(0)
def get_frame(self):
BaseCamera.last_access = time.time()
BaseCamera.event.wait()
BaseCamera.event.clear()
return BaseCamera.frame
@staticmethod
def frames():
raise RuntimeError('Must be implemented by subclasses.')
@classmethod
def _thread(cls):
print('Starting camera thread.')
frames_iterator = cls.frames()
for frame in frames_iterator:
BaseCamera.frame = frame
BaseCamera.event.set()
time.sleep(0)
if time.time() - BaseCamera.last_access > 10:
frames_iterator.close()
print('Stopping camera thread due to inactivity.')
break
BaseCamera.thread = None
class Camera(BaseCamera):
video_source = 0
@staticmethod
def set_video_source(source):
Camera.video_source = source
@staticmethod
def frames():
camera = cv2.VideoCapture(Camera.video_source)
if not camera.isOpened():
raise RuntimeError('Error')
while True:
image = ImageGrab.grab() # 获取屏幕数据
# w, h = image.size
image = image.resize((1366, 750)) # 图片缩放
output_buffer = BytesIO() # 创建二进制对象
image.save(output_buffer, format='JPEG', quality=100) # quality提升图片分辨率
frame = output_buffer.getvalue() # 获取二进制数据
yield frame # 生成器返回一张图片的二进制数据
app = Flask(__name__)
@app.route('/')
def index():
"""
视图函数
:return:
"""
return render_template_string('''<html>
<head>
<title>屏幕共享</title>
</head>
<body>
<img src="{{ url_for('video_feed') }}">
</body>
</html>''')
def gen(camera):
"""
流媒体发生器
"""
while True:
frame = camera.get_frame()
yield (b'--frame\r\n'
b'Content-Type: image/jpeg\r\n\r\n' + frame + b'\r\n')
@app.route('/video_feed')
def video_feed():
"""流媒体数据"""
return Response(gen(Camera()),
mimetype='multipart/x-mixed-replace; boundary=frame')
if __name__ == '__main__':
ip_host = '127.0.0.1'# 本机ip地址
ip_host2 = '0.0.0.0'# 内网ip地址
app.run(threaded=True, host=ip_host2, port=80)
processing.py
import cv2
url_path = "http://192.168.7.70/video_feed"
cap = cv2.VideoCapture(url_path)
while True:
ret, frame = cap.read()
if not ret:
break
cv2.imshow("frame", frame)
cv2.waitKey(1)
cap.release()
cv2.destroyAllWindows()