ROS2--随手笔记

marcker - - mash type

marker = visualization_msgs.msg.Marker()
marker.type = visualization_msgs.msg.Marker.MESH_RESOURCE
marker.mesh_resource = 'package://HexagonalPrism.STL'
//解释
文件格式为 'package://' + name_of_package + '/' + filename_in_package
例如,如果您在hexbot软件包中包含HexagonalPrism.STL,则语法应为'package://hexbot/HexagonalPrism.STL'。如果它在该程序包的一个名为meshes的文件夹中,'package://hexbot/meshes/HexagonalPrism.STL'

注:可以根据启动RViz加载Macker后报的错来修改加载文件的路径

[ERROR] [rviz2]: Error retrieving file [file:///home/gw04/all_ws/ros2_tt/install/data/share/data/generic_suv_1.stl]: Couldn't open file /home/gw04/all_ws/ros2_tt/install/data/share/data/generic_suv_1.stl

自定义的msg

float64类型等同于double类型
float64[ ]  ppp  #这里的类型是vector在使用ppp时可以使用ppp.push_back(***)

CMakeLists.txt中添加pcl


cmake_minimum_required(VERSION 3.5)
project(point_display)

# Default to C99
if(NOT CMAKE_C_STANDARD)
  set(CMAKE_C_STANDARD 99)
endif()

# Default to C++14
if(NOT CMAKE_CXX_STANDARD)
  set(CMAKE_CXX_STANDARD 14)
endif()

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(my_msgs REQUIRED)
#find_package(pcl_ros REQUIRED)
find_package(pcl_conversions REQUIRED)
find_package(sensor_msgs REQUIRED)

 find_package(PCL 1.8 REQUIRED)
 include_directories(${PCL_INCLUDE_DIRS})
 link_directories(${PCL_LIBRARY_DIRS})
 add_definitions(${PCL_DEFINITIONS})
find_package(Boost REQUIRED)
 include_directories(${Boost_INCLUDE_DIRS})
 link_directories(${Boost_LIBRARY_DIRS})
#find_package(pcl_ros REQUIRED)
#find_package(pcl_conversions REQUIRED)

# uncomment the following section in order to fill in
# further dependencies manually.
# find_package(<dependency> REQUIRED)

if(BUILD_TESTING)
  find_package(ament_lint_auto REQUIRED)
  # the following line skips the linter which checks for copyrights
  # uncomment the line when a copyright and license is not present in all source files
  set(ament_cmake_copyright_FOUND TRUE)
  # the following line skips cpplint (only works in a git repo)
  # uncomment the line when this package is not in a git repo
  set(ament_cmake_cpplint_FOUND TRUE)
  ament_lint_auto_find_test_dependencies()
endif()
##add
include_directories(include
  ${libpcap_LIBRARIES}
)
##
 add_executable(points_display src/points_display.cpp)
ament_target_dependencies(points_display rclcpp std_msgs sensor_msgs mymsgs pcl_conversions)
//这里的ament_target_dependencies()将需要的消息类型连接到目标文件
target_link_libraries(points_display ${PCL_LIBRARIES})
//注意这里是target_link_libraries(),把pcl的库连接到目标文件

install(TARGETS
  points_display
  DESTINATION lib/${PROJECT_NAME}
)
ament_package()

动态外部修改parameters

  • 通过ros2 param set [node_name] [param_name] [param_value]
  • 节点param_test
#include <iostream>
#include <memory>

#include "rclcpp/rclcpp.hpp"

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

  rclcpp::Node::SharedPtr node = rclcpp::Node::make_shared("param_test");

  // TODO(wjwwood): Make the parameter service automatically start with the node.
  //rclcpp::ParameterService::SharedPtr parameter_service = std::make_shared<rclcpp::ParameterService>(node);

  rclcpp::SyncParametersClient::SharedPtr parameters_client = std::make_shared<rclcpp::SyncParametersClient>(node);
//declare parameters
node->declare_parameter("foo");
node->declare_parameter("bar");
node->declare_parameter("baz");
node->declare_parameter("foobar");
  // Set several different types of parameters.
  auto set_parameters_results = parameters_client->set_parameters({
    rclcpp::Parameter("foo", 2),
    rclcpp::Parameter("bar", "hello"),
    rclcpp::Parameter("baz", 1.45),
    rclcpp::Parameter("foobar", true),
  });
  // Check to see if they were set.
  for (auto & result : set_parameters_results) {
    if (!result.successful) {
      std::cerr << "Failed to set parameter: " << result.reason << std::endl;
    }
  }
  std::string tt;
while(1)
{
  // Get a few of the parameters just set.
  for (auto & parameter : parameters_client->get_parameters({"foo", "bar" ,"baz" ,"foobar"})) {
    std::cout << "Parameter name: " << parameter.get_name() << std::endl;
    std::cout << "Parameter value (" << parameter.get_type_name() << "): " <<
      parameter.value_to_string() << std::endl;
  }
  //也可以这么使用
  auto tt = parameters_client->get_parameters({"foo", "bar" ,"baz" ,"foobar"});
  //auto tt1 = tt[0].value_to_string();
  //auto tt1 = tt[0].as_int();
auto tt2 = tt[0].get_type_name();
auto tt2_ = tt[0].as_int();
auto tt3 = tt[1].get_type_name();
auto tt3_ = tt[1].as_string();
auto tt4 = tt[2].get_type_name();
auto tt4_ = tt[2].as_double();
auto tt5 = tt[3].get_type_name();
auto tt5_ = tt[3].as_bool();
//std::cout << "Parameter value: " << tt1 << std::endl;
std::cout << "Parameter value type: " << tt2 <<" "<<tt3<<" "<<tt4<<" "<<tt5<< std::endl;
std::cout << "Parameter value     : " << tt2_ <<"  "<<tt3_<<"  "<<tt4_<<"  "<<tt5_<< std::endl;
}
  return 0;
}
  • 注意的语句:
// 1 建立参数客户端
  rclcpp::SyncParametersClient::SharedPtr parameters_client = std::make_shared<rclcpp::SyncParametersClient>(node);
 // 2 声明参数
node->declare_parameter("foo");
node->declare_parameter("bar");
node->declare_parameter("baz");
node->declare_parameter("foobar");
// 3 设置不同类型的参数
  auto set_parameters_results = parameters_client->set_parameters({
    rclcpp::Parameter("foo", 2),
    rclcpp::Parameter("bar", "hello"),
    rclcpp::Parameter("baz", 1.45),
    rclcpp::Parameter("foobar", true),
  });
  • 如果需要参数有默认值可以使用下面的两种方法(使用node的成员函数)
//方法1:声明时初始化
node->declare_parameter("pub_topic", std::string("/display/left"));
//方法2:设置时获取时设置
string sub_topic;
node->get_parameter_or_set("sub_topic", sub_topic, std::string("/corner/data/front/left"));

注:

可以向上面那样使用来设置参数是因为,当节点启动时,参数服务器就启动了:

自我理解1// 在node_ptions.hpp中有下面的注释,其中“- start_parameter_services = true”
  /// Create NodeOptions with default values, optionally specifying the allocator to use.
  /**
   * Default values for the node options:
   *
   *   - context = rclcpp::contexts::default_context::get_global_default_context()
   *   - arguments = {}
   *   - parameter_overrides = {}
   *   - use_global_arguments = true
   *   - use_intra_process_comms = false
   *   - start_parameter_services = true
   *   - start_parameter_event_publisher = true
   *   - parameter_event_qos = rclcpp::ParameterEventQoS
   *     - with history setting and depth from rmw_qos_profile_parameter_events
   *   - parameter_event_publisher_options = rclcpp::PublisherOptionsBase
   *   - allow_undeclared_parameters = false
   *   - automatically_declare_parameters_from_overrides = false
   *   - allocator = rcl_get_default_allocator()
   *
   * \param[in] allocator allocator to use in construction of NodeOptions.
   */
   自我理解2:
   将param_test.cpp文件中除了定义初始化node的语句都注释掉,return 0;前加上while(1);,在编译运行;
   新打开终端:ros2 service list  
   输出:
   xx@xx:~$ ros2 service list
/param_test/describe_parameters
/param_test/get_parameter_types
/param_test/get_parameters
/param_test/list_parameters
/param_test/set_parameters
/param_test/set_parameters_atomically
这说明定义node,启动节点后parameter service也启动了。

launch文件启动节点,输出到屏幕

import launch  
import launch_ros.actions
def generate_launch_description():
  node1 = launch_ros.actions.Node(
      package='ros2tt', node_executable='param_test',
      node_name='param_test',
      parameters=[{'foo': 1}])
  return launch.LaunchDescription([node1,])
  • 这时通过ros2 launch ‘launch文件路径’ 启动节点时显示:
[INFO] [launch]: Default logging verbosity is set to INFO
  • 若要是输出打印到屏幕上,需要添加“ output=‘screen’ ”
import launch  
import launch_ros.actions
def generate_launch_description():
  node1 = launch_ros.actions.Node(
      package='ros2tt', node_executable='param_test',
      node_name='param_test', output='screen',
      parameters=[{'foo': 1}])
  return launch.LaunchDescription([node1,])

此时,程序里的std::cout可以显示在屏幕上
也可以使用:

RCLCPP_INFO(node->get_logger(), "I heard :=%f",rclcpp::Duration(0,40000000).seconds());

进行输出显示。

  • 可以对log进行宏定义使用
#include <rcutils/logging_macros.h>
#define ROS_ERROR RCUTILS_LOG_ERROR
#define ROS_INFO RCUTILS_LOG_INFO
#define ROS_ERROR_THROTTLE(sec, ...) RCUTILS_LOG_ERROR_THROTTLE(RCUTILS_STEADY_TIME, sec, __VA_ARGS__) //待查明使用方法
----------------
使用时:
 ROS_ERROR("Open Serial: %s Error!",serialNumber_.c_str());
 ROS_INFO("Open serial: [ %s ], successfully, with idex: %d.", serialNumber_.c_str() ,fd); -------------------
参考:https://blog.csdn.net/u011118482/article/details/80389767

自己构建tf::StampedTransform的函数

ROS1中的tf包含tf::StampedTransform的函数,ROS2中没有这个函数,由于需要使用这种操作,所以进行了构建

//使用到的
#include "rclcpp/rclcpp.hpp"
#include <tf2_ros/transform_broadcaster.h>
#include <tf2_ros/static_transform_broadcaster.h>
#include <tf2/utils.h> 
#include <tf2/transform_datatypes.h>
#include <tf2/LinearMath/Quaternion.h>
//头文件

geometry_msgs::msg::TransformStamped StampedTransform_self(const tf2::Transform& transform_, const rclcpp::Time& timestamp, const std::string & frame_id, const std::string & child_frame_id)
{
	  std::cout<<"--1-StampedTransform_self--"<<std::endl;
   tf2::Quaternion q_tf2;
   tf2::Vector3 v_tf2;
   geometry_msgs::msg::Quaternion q_geo;
   geometry_msgs::msg::Vector3    t_geo;
   
   v_tf2 = transform_.getOrigin();
   t_geo.set__x(v_tf2.getX());
   t_geo.set__y(v_tf2.getY());
   t_geo.set__z(v_tf2.getZ());
   /*
   std::cout<<"v_tf2.getX() = "<<v_tf2.getX() <<std::endl;
   std::cout<<"v_tf2.getY() = "<<v_tf2.getY() <<std::endl;
   std::cout<<"v_tf2.getZ() = "<<v_tf2.getZ() <<std::endl;
   */
    q_tf2 = transform_.getRotation();
    q_geo.set__x(q_tf2.getX());
    q_geo.set__y(q_tf2.getY());
    q_geo.set__z(q_tf2.getZ());
    q_geo.set__w(q_tf2.getW());
    /*
   std::cout<<"q_tf2.getX() = "<<q_tf2.getX() <<std::endl;
   std::cout<<"q_tf2.getY() = "<<q_tf2.getY() <<std::endl;
   std::cout<<"q_tf2.getZ() = "<<q_tf2.getZ() <<std::endl;
   std::cout<<"q_tf2.getW() = "<<q_tf2.getW() <<std::endl;
    */
    geometry_msgs::msg::TransformStamped geo_trans;
	  geo_trans.header.stamp = timestamp;
	  geo_trans.header.frame_id = frame_id;//"velodyne";
	  geo_trans.child_frame_id = child_frame_id;//"body";
	  geo_trans.transform.translation = t_geo;
	  geo_trans.transform.rotation = q_geo;
    std::cout<<"--2-StampedTransform_self--"<<std::endl;
    return geo_trans;
}

从众多坑中爬出来后发现:

https://github.com/zhangrelay/ros2_tutorials
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: ROS2 Web Bridge是一个开源的软件包,旨在使ROS2(Robot Operating System 2)与Web端进行通信和交互。它提供了一种简单而强大的方式,通过WebSocket协议将ROS2系统中的数据传输到Web浏览器。 ROS2 Web Bridge允许Web开发人员使用常见的Web技术(例如JavaScript)直接与ROS2系统进行交互。它提供了一个轻量级的接口,可以订阅和发布ROS2主题,并在Web浏览器中实时显示传感器数据、控制机器人等。 此外,ROS2 Web Bridge还支持ROS2服务和动作。它允许Web应用程序在Web端调用ROS2服务,从而实现与ROS2节点的双向通信。通过一个用户友好的Web界面,用户可以发送控制命令给机器人,执行任务并获得实时反馈。 ROS2 Web Bridge的主要优点之一是其跨平台性。它基于WebSocket协议,因此可以在不同的操作系统和设备上使用,无论是在PC端还是移动设备上。 此外,ROS2 Web Bridge还支持认证和授权机制,以确保通信安全。这对于确保只有被授权的用户可以访问和控制ROS2系统非常重要。 总的来说,ROS2 Web Bridge为ROS2系统提供了一种简便而强大的方式,使Web开发人员能够与ROS2系统进行交互。它扩展了ROS2的功能,使得机器人开发更加灵活和可视化。 ### 回答2: ros2-web-bridge是一种用于在ROS 2和Web应用程序之间进行通信的桥接工具。ROS 2是机器人操作系统的第二代版本,而Web应用程序是通过Web浏览器访问的应用程序。 ros2-web-bridge有几个主要功能。首先,它允许ROS 2中的节点直接与通过Web浏览器访问的Web应用程序进行通信。这使得在Web界面上实时监控和控制ROS 2系统变得更加容易。例如,可以使用ros2-web-bridge将传感器数据从ROS 2节点发送到Web应用程序,以便在Web界面上实时显示传感器数据。同时,还可以将来自Web界面的用户输入发送到ROS 2节点,以便远程控制机器人或系统。 其次,ros2-web-bridge还提供了一些工具和API,用于将ROS 2中的消息和服务转换为Web格式。这使得可以轻松地在ROS 2和Web应用程序之间进行数据传输和交互。它支持使用ROS 2的套接字和JSON进行通信,并提供了将消息和服务转换为JSON格式以及反向转换的功能。这样,ROS 2节点可以与通过Web浏览器访问的Web应用程序进行无缝通信。 总而言之,ros2-web-bridge为ROS 2和Web应用程序之间的通信提供了一个便捷的桥梁。它简化了ROS 2系统与Web界面之间的集成,并提供了实时数据传输和远程控制的能力。这对于构建基于ROS 2的机器人或系统的开发者和使用者来说是非常方便的工具。 ### 回答3: ros2-web-bridge是一个用于将ROS 2和Web技术进行连接的工具。它提供了一个桥接器,使得可以通过Web浏览器与ROS 2通信。这个工具是建立在ROS 2和Web Socket之间的通信基础上的。 通过ros2-web-bridge,我们可以在Web浏览器中实时地订阅和发布ROS 2的消息。这使得我们可以通过Web界面来控制ROS 2的机器人,或者将ROS 2的数据可视化展示出来。这对于远程监控、远程操作和数据可视化都非常有用。 ros2-web-bridge使用ROS 2提供的接口来与ROS 2系统进行通信。它将ROS 2的消息转换为适用于Web Socket的格式,并在浏览器和ROS 2之间建立起适配的连接。通过这种方式,Web界面就能够与ROS 2系统进行实时的双向通信。 ROS 2中的消息传递方式是异步的,而Web浏览器通常使用同步的方式进行通信。ros2-web-bridge通过在ROS 2和Web Socket之间进行适配和转换,使得二者能够协同工作。这意味着ROS 2中的数据可以通过ros2-web-bridge传输到Web浏览器,并在该浏览器中进行处理和展示。 总的来说,ros2-web-bridge是一个功能强大的工具,它架起了ROS 2和Web技术之间的桥梁。它使得我们可以通过Web浏览器来与ROS 2进行通信,进而实现远程操作和数据可视化的目的。通过ros2-web-bridge,我们可以更加灵活和方便地利用ROS 2的功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值