PIBOT移植ROS2记录(5)-参数及动态设置

1. pid参数调节的topic

1.1 移植

pid的输入输出作为topic分别输出,这个只需要创建一个int类型的pub即可

void BaseDriver::init_pid_debug() {
  if (config_.out_pid_debug_enable) {
    const char* input_topic_name[MAX_MOTOR_COUNT] = {"motor1_input", "motor2_input", "motor3_input", "motor4_input"};
    const char* output_topic_name[MAX_MOTOR_COUNT] = {"motor1_output", "motor2_output", "motor3_output", "motor4_output"};
    for (size_t i = 0; i < MAX_MOTOR_COUNT; i++) {
      pid_input_pub_[i] = this->create_publisher<std_msgs::msg::Int32>(input_topic_name[i], 10);
      pid_output_pub_[i] = this->create_publisher<std_msgs::msg::Int32>(output_topic_name[i], 10);
    }
  }
}

void BaseDriver::update_pid_debug() {
  if (config_.out_pid_debug_enable) {
    frame_->interact(ID_GET_PID_DATA);

    for (size_t i = 0; i < MAX_MOTOR_COUNT; i++) {
      pid_input_msg_[i].data = DataHolder::get()->pid_data.input[i];
      pid_output_msg_[i].data = DataHolder::get()->pid_data.output[i];

      pid_input_pub_[i]->publish(pid_input_msg_[i]);
      pid_output_pub_[i]->publish(pid_output_msg_[i]);
    }
  }
}

1.2 运行测试

在launch文件中打开out_pid_debug_enable后启动

...
{"out_pid_debug_enable": True},
ros2 launch  pibot_bringup bringup_launch.py

查看topic列表

➜  ros2 topic list
/cmd_vel
/motor1_input
/motor1_output
/motor2_input
/motor2_output
/motor3_input
/motor3_output
/motor4_input
/motor4_output
/odom
/parameter_events
/rosout
/tf

1.3 rqt plot

安装rqt-plot

sudo apt-get install ros-galactic-rqt-plot

安装后rqt未发现plot的插件,原因暂时位置

2. 动态参数

2.1 移植

不同于ROS1中的dynamic_reconfigure,ROS2中我们直接使用declare_parameter声明参数,可以在rqt-reconfigure中动态配置,之前我们在声明时新增了一个只读的约束

这里我们还可以新增其他约束以限制参数设置的范围


    rcl_interfaces::msg::ParameterDescriptor descriptor;
    descriptor.description = "";
    descriptor.name = "name";
    descriptor.integer_range.resize(1);
    descriptor.integer_range[0].from_value = 10;
    descriptor.integer_range[0].to_value = 1000;
    descriptor.integer_range[0].step = 1;
    node->declare_parameter<uint16_t>("wheel_diameter", rp->wheel_diameter, descriptor);

同时我们设置一个参数修改的回调通知,来根据设置的参数下发至下位机

  rclcpp::Node::OnSetParametersCallbackHandle::SharedPtr callback_handle_;
    callback_handle_ = node->add_on_set_parameters_callback(std::bind(&BaseDriverConfig::SetParametersCallback,
                                                                      this,
                                                                      std::placeholders::_1,
                                                                      node,
                                                                      rp));

回调中我们需要循环所有参数列表,并且判断参数名称设置相应的变量


rcl_interfaces::msg::SetParametersResult BaseDriverConfig::SetParametersCallback(const std::vector<rclcpp::Parameter>& parameters, rclcpp::Node* node, Robot_parameter* rp) {
  rcl_interfaces::msg::SetParametersResult result;
  result.successful = true;
  result.reason = "success";
  for (auto& param : parameters) {
    RCLCPP_INFO(node->get_logger(), "param %s update", param.get_name().c_str());
    if (param.get_name() == "motor1_exchange_flag") {
      RCLCPP_INFO(node->get_logger(), "++param %d", rp->motor_nonexchange_flag);
      std::bitset<8> val(rp->motor_nonexchange_flag);
      val[0] = !param.as_bool();
      rp->motor_nonexchange_flag = val.to_ulong();
      RCLCPP_INFO(node->get_logger(), "--param %d", rp->motor_nonexchange_flag);
    } else if (param.get_name() == "motor2_exchange_flag") {
      std::bitset<8> val(rp->motor_nonexchange_flag);
      val[1] = !param.as_bool();
      rp->motor_nonexchange_flag = val.to_ulong();
    } else if (param.get_name() == "motor3_exchange_flag") {
      std::bitset<8> val(rp->motor_nonexchange_flag);
      val[2] = !param.as_bool();
      rp->motor_nonexchange_flag = val.to_ulong();
    } else if (param.get_name() == "motor4_exchange_flag") {
      std::bitset<8> val(rp->motor_nonexchange_flag);
      val[3] = !param.as_bool();
      rp->motor_nonexchange_flag = val.to_ulong();
    } else if (param.get_name() == "encoder1_exchange_flag") {
      std::bitset<8> val(rp->encoder_nonexchange_flag);
      val[0] = !param.as_bool();
      rp->encoder_nonexchange_flag = val.to_ulong();
    } else if (param.get_name() == "encoder2_exchange_flag") {
      std::bitset<8> val(rp->encoder_nonexchange_flag);
      val[1] = !param.as_bool();
      rp->encoder_nonexchange_flag = val.to_ulong();
    } else if (param.get_name() == "encoder3_exchange_flag") {
      std::bitset<8> val(rp->encoder_nonexchange_flag);
      val[2] = !param.as_bool();
      rp->encoder_nonexchange_flag = val.to_ulong();
    } else if (param.get_name() == "encoder4_exchange_flag") {
      std::bitset<8> val(rp->encoder_nonexchange_flag);
      val[3] = !param.as_bool();
      rp->encoder_nonexchange_flag = val.to_ulong();
    } else if (param.get_name() == "model_type") {
      rp->model_type = param.as_int();
    } else if (param.get_name() == "wheel_diameter") {
      rp->wheel_diameter = param.as_int();
    } else if (param.get_name() == "wheel_track") {
      rp->wheel_track = param.as_int();
    } else if (param.get_name() == "encoder_resolution") {
      rp->encoder_resolution = param.as_int();
    } else if (param.get_name() == "do_pid_interval") {
      rp->do_pid_interval = param.as_int();
    } else if (param.get_name() == "kp") {
      rp->kp = param.as_int();
    } else if (param.get_name() == "ki") {
      rp->ki = param.as_int();
    } else if (param.get_name() == "kd") {
      rp->kd = param.as_int();
    } else if (param.get_name() == "ko") {
      rp->ko = param.as_int();
    } else if (param.get_name() == "cmd_last_time") {
      rp->cmd_last_time = param.as_int();
    } else if (param.get_name() == "max_v_liner_x") {
      rp->max_v_liner_x = param.as_int();
    } else if (param.get_name() == "max_v_liner_y") {
      rp->max_v_liner_y = param.as_int();
    } else if (param.get_name() == "max_v_angular_z") {
      rp->max_v_angular_z = param.as_int();
    } else if (param.get_name() == "imu_type") {
      rp->imu_type = param.as_int();
    } else if (param.get_name() == "motor_ratio") {
      rp->motor_ratio = param.as_int();
    }
  }

  DataHolder::dump_params(rp);

  param_update_flag_ = true;
  return result;
}

该回调被调用会设置一个update_flag的变量,主线程会处理执行一次参数同步操作

2.2 运行测试

ros2 launch  pibot_bringup bringup_launch.py
ros2 run rqt_reconfigure rqt_reconfigure

不同于ROS1的dynamic_reconfigure, 显示的参数不会按照我们声明的顺序,而是按照字母排序,会显得有点杂乱。
参数

同时暂时无法配置枚举类型,对于之前的model_type,在ui中无法下拉列表选择,略显麻烦

本文代码https://gitee.com/pibot/pibot_bringup/tree/25ed34acda8a6e850a1c96fcee8d3a762374a135

  • 1
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答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的功能。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值