【机器人仿真Webots教程】-控制器编程指南

这篇具有很好参考价值的文章主要介绍了【机器人仿真Webots教程】-控制器编程指南。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

Webots controller编程指南

1. controller程序设计流程

1.1 controller与场景树节点

在Webots中,场景树节点(Scene Tree Nodes)是Webots仿真环境中的各种对象,包括机器人模型、传感器、环境物体等。每个节点都有其在场景树中的位置,节点之间可以形成层次结构,以便组织和管理模拟环境。

控制器程序(Controller Program)是用于控制机器人在仿真环境中执行动作和决策的代码。它可以是任何支持Webots API的编程语言,例如C、C++、Python等。控制器程序可以访问和操纵场景树中的节点,以实现机器人的行为。

【机器人仿真Webots教程】-控制器编程指南

关联场景树节点和控制器程序的过程通常如下:

  1. 创建或导入机器人模型和其他物体,这些将成为场景树节点的一部分。
  2. 在Webots中创建一个控制器程序文件,并将其与一个或多个场景树节点关联。关联可以通过以下方式之一完成:
    • 使用Webots图形用户界面(GUI):选择一个或多个节点,然后在GUI中设置它们的控制器程序路径。
    • 使用Webots API:在控制器程序中使用适当的API函数来获取对场景树节点的引用,并将其与控制器程序关联起来。
  3. 编写控制器程序,使用适当的API函数来操作和控制场景树节点。这包括读取传感器数据、发送控制命令、更新节点状态等。
  4. 在Webots中运行仿真,控制器程序将与场景树节点进行交互,实现机器人的控制和行为。

通过这种方式,控制器程序可以与场景树节点进行关联,并利用节点的属性和功能来控制仿真环境中的机器人行为。

【机器人仿真Webots教程】-控制器编程指南

1.2 进程角度分析

从进程的角度观察,每个控制器进程都是webots进程的子进程,同时每个控制器进程和webots进程并不共享内存(除Camera外)

【机器人仿真Webots教程】-控制器编程指南

1.3 仿真时间步长与控制器程序更新延时步长

step指的是仿真步长,这个步长是整个仿真环境(具体来说是场景树)中更新计算的时间间隔,在场景树中指定WorldInfo.basicTimeStep

wb_robot_step是controller程序的更新间隔,一个wb_robot_step包含若干step(simulation step)

例如仿真步长为16ms,那么wb_robot_step可以是16、32、64、128ms,单击界面中的step按钮,即走一个simulation step,如果wb_robot_step包含多个simulation step,那么会打断wb_robot_step的执行。

【机器人仿真Webots教程】-控制器编程指南

2. 例解控制器编程

2.1 Hello world 实例

wb_robot_init 为C Api在函数调用之前调用的初始化函数,初始化controller和webots lib的通信,在C API中特有,其他编程语言中并不存在

wb_robot_cleanup 做善后工作,在C API中特有,其他编程语言中并不存在。

wb_robot_step 在每个controller中都有,定期调用,因此通常放在主循环中,参数代表毫秒数,,为控制步骤的持续时间,函数根据设定的毫秒数进行仿真返回计算值,仿真时间量并不是真实的时间,在实际中可能是1毫秒或者一分钟的时间。注意,webots终止时返回wb_robot_step -1。只要仿真进行,那么控制回路就会运行

Hello world入门实例如下:

#include <webots/robot.h>
#include <stdio.h>

int main() {
  wb_robot_init();

  while(wb_robot_step(32) != -1)
    printf("Hello World!\n");

  wb_robot_cleanup();
  return 0;
}

2.2 传感器读取实例

#include <webots/robot.h>
#include <webots/distance_sensor.h>
#include <stdio.h>

#define TIME_STEP 32

int main() {
  wb_robot_init();

  WbDeviceTag sensor = wb_robot_get_device("my_distance_sensor");
  wb_distance_sensor_enable(sensor, TIME_STEP);

  while (wb_robot_step(TIME_STEP) != -1) {
    const double value = wb_distance_sensor_get_value(sensor);
    printf("Sensor value is %f\n", value);
  }

  wb_robot_cleanup();
  return 0;
}

使用设备需要获取设备的标签,类型为WbDeviceTag,获取方式为wb_robot_get_device,参数为机器人描述文件(.wbt或者.proto文件)中的设备名称,获取失败返回0

传感器设备在使用之前需要调用wb_*_enable进行使能,参数为获取的WbDeviceTag类型的变量(指定使能的设备)以及传感器数据两次更新之间的延时时间(TIME_STEP),一般的此处设置的延时时间与wb_robot_step的参数有关,一般为倍数关系,例如延时时间设置为控制步长的两倍,那么传感器每两次wb_robot_step调用更新一次,如果设置的延时时间比控制步长还要短是没有意义的,可以理解wb_robot_step的控制步长为一个最小单位时间。

wb_*_disable禁用设备,这样有可能会提高仿真的速度。wb_distance_sensor_get_value函数可以获取距离传感器的最新值,那么,类似的,还有很多获取传感器值的API,但是有可能这些设备返回的是个数组,例如加速度计、GPS、陀螺仪等等,接收数据如下:

API原型

const double *wb_gps_get_values(WbDeviceTag tag);
const double *wb_accelerometer_get_values(WbDeviceTag tag);
const double *wb_gyro_get_values(WbDeviceTag tag);

获取

const double *values = wb_gps_get_values(gps);

// OK, to read the values they should never be explicitly deleted by the controller code
printf("MY_ROBOT is at position: %g %g %g\n", values[0], values[1], values[2]);

// OK, to copy the values
double x, y, z;
x = values[0];
y = values[1];
z = values[2];

2.3 执行器的使用

controller也需要WbDeviceTag类型的变量作为入参指定设备,其不需要使能设备

电机控制实例

#include <webots/robot.h>
#include <webots/motor.h>
#include <math.h>

#define TIME_STEP 32

int main() {
  wb_robot_init();

  WbDeviceTag motor = wb_robot_get_device("my_motor");

  const double F = 2.0;   // frequency 2 Hz
  double t = 0.0;         // elapsed simulation time

  while (wb_robot_step(TIME_STEP) != -1) {
    const double position = sin(t * 2.0 * M_PI * F);
    wb_motor_set_position(motor, position);
    t += (double)TIME_STEP / 1000.0;
  }

  wb_robot_cleanup();
  return 0;
}

功能说明:使旋转电机以2Hz正弦信号震荡.

API解释

函数名 函数功能 参数
wb_motor_set_position 设置旋转电机位置 设备描述符,位置量

wb_motor_set_position设置位置之后并不会立即启动电机,而是等待wb_robot_step的驱动,将驱动命令发送到RationalMotor中,在指定的控制步长的时间(单位毫秒)内仿真电机的运动,一个控制步长的时间并不一定可以完成整个运动的过程。

#include <webots/robot.h>
#include <webots/motor.h>
#include <math.h>

#define TIME_STEP 32

int main() {
  wb_robot_init();

  WbDeviceTag motor = wb_robot_get_device("my_motor");

  const double F = 2.0;   // frequency 2 Hz
  double t = 0.0;         // elapsed simulation time

  while (wb_robot_step(TIME_STEP) != -1) {
    const double position = sin(t * 2.0 * M_PI * F);
    wb_motor_set_position(motor, position);
    t += (double)TIME_STEP / 1000.0;
  }

  wb_robot_cleanup();
  return 0;
}

一般情况下为控制运动的行进建模,将整个运动分解为离散的组合步骤,一般这个离散步骤的单位就是wb_robot_step的参数。

2.4 传感器与执行器综合例程

功能描述:机器人使用的是差动转向。它使用两个距离传感器 ( DistanceSensor) 来检测障碍物。

#include <webots/robot.h>
#include <webots/motor.h>
#include <webots/distance_sensor.h>

#define TIME_STEP 32

int main() {
  wb_robot_init();

  WbDeviceTag left_sensor = wb_robot_get_device("left_sensor");
  WbDeviceTag right_sensor = wb_robot_get_device("right_sensor");
  wb_distance_sensor_enable(left_sensor, TIME_STEP);
  wb_distance_sensor_enable(right_sensor, TIME_STEP);

  WbDeviceTag left_motor = wb_robot_get_device("left_motor");
  WbDeviceTag right_motor = wb_robot_get_device("right_motor");
  wb_motor_set_position(left_motor, INFINITY);
  wb_motor_set_position(right_motor, INFINITY);
  wb_motor_set_velocity(left_motor, 0.0);
  wb_motor_set_velocity(right_motor, 0.0);

  while (wb_robot_step(TIME_STEP) != -1) {

    // read sensors
    const double left_dist = wb_distance_sensor_get_value(left_sensor);
    const double right_dist = wb_distance_sensor_get_value(right_sensor);

    // compute behavior (user functions)
    const double left = compute_left_speed(left_dist, right_dist);
    const double right = compute_right_speed(left_dist, right_dist);

    // actuate wheel motors
    wb_motor_set_velocity(left_motor, left);
    wb_motor_set_velocity(right_motor, right);
  }

  wb_robot_cleanup();
  return 0;
}

注意,更新传感器状态一定要通过wb_robot_step,因为在wb_robot_step之前,API仅仅做了预备工作,并不会直接对仿真的进程进行推进(也即设备状态不会立即得到响应),只有调用了wb_robot_step才会推进仿真进程的响应。

2.5 控制器读入参数、程序终止

对于控制器的参数,通过控制器的main函数入口的argv接收,指定方式为机器人的controllerArgs节点字段

例:

参数:

Robot {
  ...
  controllerArgs "one two three"
  ...
}

控制器(名为demo)程序:

#include <webots/robot.h>
#include <stdio.h>

int main(int argc, const char *argv[]) {
  wb_robot_init();

  int i;
  for (i = 0; i < argc; i++)
    printf("argv[%i]=%s\n", i, argv[i]);

  wb_robot_cleanup();
  return 0;
}

输出:

argv[0]=demo
argv[1]=one
argv[2]=two
argv[3]=three

一般控制器的主程序程序是一个大循环,控制器的终止也是循环的终止。导致控制器发生终止的事件一般有:

  • Webots退出
  • 仿真重置
  • world重新加载
  • 加载新的仿真
  • 控制器名称修改

当上述事件发生时wb_robot_step返回-1,控制器进程不再与webots进程通信,在实际时间的1s后,如果控制器程序没有主动终止,Webots进行将发送SIGKILL信号杀死控制器进程,并给控制器程序足够的时间完成数据的转储和文件关闭。

示例:

功能描述:控制器程序终止之前保存数据。

#include <webots/robot.h>
#include <webots/distance_sensor.h>
#include <stdio.h>

#define TIME_STEP 32

int main() {
  wb_robot_init();

  WbDeviceTag sensor = wb_robot_get_device("my_distance_sensor");
  wb_distance_sensor_enable(sensor, TIME_STEP);

  while (wb_robot_step(TIME_STEP) != -1) {
    const double value = wb_distance_sensor_get_value();
    printf("sensor value is %f\n", value);
  }

  // Webots triggered termination detected!
  // Past this point, new printf statements will no longer be
  // displayed in the Webots console

  saveExperimentData();  // this shouldn't last longer than one second

  wb_robot_cleanup();
  return 0;
}

在某些情况下,由控制器做出终止模拟的决定。例如在搜索和优化算法的情况下:搜索可能会在找到解决方案时或在固定次数的迭代(或生成)之后终止。

在这种情况下,控制器应该只保存实验结果并通过从函数返回main或调用exit函数退出。这将终止控制器进程并在当前仿真步骤冻结仿真。物理仿真和仿真中涉及的每个机器人都将停止。

2.6 控制台

控制器程序支持stdout、stderr,并被程序重定向到了webots的控制台,webots不支持stdin(即标准输入),仅支持一些ANSI转义码进行文本样式的设置和清除,支持类型有以下所示:

  • 3 位色彩(前景和背景)
  • Blob 风格
  • 下划线 风格
  • clear清屏
  • 重置(颜色和样式)

示例程序:

world文件:

WEBOTS_HOME/projects/samples/howto/console/worlds/console.wbt

控制器文件:

WEBOTS_HOME/projects/samples/howto/console/controllers/console/console.c

ANSI头文件:

WEBOTS_HOME/include/controller/c/webots/utils/ansi_codes.h

#include <webots/utils/ansi_codes.h>

printf("This is %sred%s!\n", ANSI_RED_FOREGROUND, ANSI_RESET);

2.7 共享库&环境变量

共享库对于控制器程序和插件之间的代码共享非常有用

  • 可以将共享库放入libraries子目录中

  • WEBOTS_HOME/resources/Makefile.include,手动修改链接共享库
    示例: WEBOTS_HOME/resources/projects/libraries/qt_utils

  • 将共享库添加到环境变量[[DY]LD_LIBRARY_]PATH

每个控制器程序目录下支持一个配置文件的方式自动在运行前加载环境变量到当前环境下,该配置文件名为"runtime.ini",runtime.ini以键值对的形式设置,包括以下7个字段:

  • [environment variables with paths]

    此部分应仅包含具有相对或绝对路径的环境变量。路径必须使用冒号“:”分隔,目录组件必须使用斜杠符号“/”分隔。本节中声明的变量将添加到每个平台上。在 Windows 上,根据 Windows 语法,冒号将替换为分号,斜线将替换为反斜线。

  • [environment variables]

    本节中定义的环境变量也将添加到每个平台的环境中,但它们将直接写入而不更改语法。对于不包含任何路径的变量来说,这是一个很好的位置。

  • [environment variables for Windows]

    如果控制器在 Windows 平台上运行,则此部分中定义的变量将仅添加到环境中。如果你想在这个部分声明路径,值应该写在双引号符号之间。

  • [environment variables for macOS]

    此处定义的变量只会在 macOS 上添加,在其他平台上会被忽略。

  • [environment variables for Linux]

    此处定义的变量将添加到所有 Linux 平台上,但不会添加到 Mac 或 Windows 上。

  • [environment variables for Linux 32]

    仅当 Linux 平台为 32 位时才会添加这些变量。

  • [environment variables for Linux 64]

    仅当 Linux 平台为 64 位时才会添加这些变量。

示例:

; typical runtime.ini

[environment variables with paths]
WEBOTS_LIBRARY_PATH = lib:$(WEBOTS_LIBRARY_PATH):../../library

[environment variables]
ROS_MASTER_URI = http://localhost:11311

[environment variables for Windows]
NAOQI_LIBRARY_FOLDER = "bin;C:\Users\My Documents\Naoqi\bin"

[environment variables for macOS]
NAOQI_LIBRARY_FOLDER = lib

[environment variables for Linux]
NAOQI_LIBRARY_FOLDER = lib

同时runtime.ini中还有关于编程语言的特定配置项,字段分别有

  • [java]
  • [python]
  • [matlab]

每个字段又包含两个键COMMANDOPTIONS,分别代表命令和选项

示例:

; runtime.ini for a Python controller on macOS

[python]
COMMAND = /opt/local/bin/python3.8
OPTIONS = -m package.name.given

上述配置等同于

/opt/local/bin/python3.8 -m package.name.given my_controller.py

示例:

; runtime.ini for a Java controller on Windows

[environment variables with paths]
CLASSPATH = ../lib/MyLibrary.jar
JAVA_LIBRARY_PATH = ../lib

[java]
COMMAND = javaw.exe
OPTIONS = -Xms6144k

注意:Java -classpath(或 - cp)选项是从CLASSPATH环境变量自动生成的。因此,您不应将其添加到OPTIONS密钥中,而应将其添加到“runtime.ini”文件中的标准环境变量中。在上面的示例中,-classpath传递给 Java 虚拟机的最终选项包括“$(WEBOTS_HOME)/lib/Controller.jar”,当前目录(“.”)或控制器 jar 文件(如果存在)(“MyController.jar”)。 jar”),最后是“…/lib/MyLibrary.jar”。文章来源地址https://www.toymoban.com/news/detail-458634.html

到了这里,关于【机器人仿真Webots教程】-控制器编程指南的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • ChatGPT +工业机器人/自动驾驶控制器的一些尝试

    ChatGPT 的功能目前已扩展到机器人领域,可以用语言直观控制如机械臂、无人机、家庭辅助机器人等的多个平台。这会改变人机交互的未来形式吗? 你可曾想过用自己的话告诉机器人该做什么,就像对人说话那样? 比如说,只要告诉你的家庭助理机器人「请帮我热一下午餐」

    2023年04月08日
    浏览(33)
  • 如何将电机控制器添加到您的 ROS 机器人

            如果您正在构建与 ROS/ROS2 一起使用的移动机器人,您需要做的第一件事就是集成电机控制器。电机控制器的目的是接受来自更高级别的软件(如导航堆栈)的消息,并将其转换为驱动电机的信号。它还将从电机的编码器接收信息,以计算机器人的速度和位置。 您

    2024年02月15日
    浏览(38)
  • 面向低成本巡线机器人的PID控制器优化——附源码

    目录 介绍 测试 电子元器件 系统特征 控制器设计 位置误差的计算 比例控制 积分控制 微分控制 改进的PID控制器 测试轨迹 源码链接 本文对经典PID控制器的改进和开环控制机制的发展进行了讨论,以提高差动轮式机器人的稳定性和鲁棒性。为了部署该算法,使用低成本、现成

    2024年04月08日
    浏览(33)
  • 基于ARM+FPGA的驱控一体机器人控制器设计

    目前市场上工业机器人,数控机床等多轴运动控制系统普遍采用运动控制器加 伺服驱动器的分布式控制方式。在这种控制方式中,控制器一方面完成人机交互,另 一方面进行 NC 代码的解释执行,插补运算,继而将计算出来的位置指令通过轴组模 块下发给各个伺服驱动器。下

    2024年02月14日
    浏览(34)
  • ABB DSQC639 3HAC025097-001/08 机器人控制器

    ABB DSQC639 3HAC025097-001/08 机器人控制器具有多种优势特性,包括: 高可靠性:该控制器采用先进的控制技术和算法,具有快速响应、高精度轨迹控制、高抗干扰能力等特点,能够满足各种复杂的应用需求。 高精度:该控制器具有高精度处理器和多种保护功能,可以确保机器人

    2024年02月04日
    浏览(32)
  • TinyMPC - CMU (卡耐基梅隆大学)开源的机器人 MPC 控制器

    CasADi - 最优控制开源 Python/MATLAB 库 TinyMPC: 资源受限微控制器上的模型预测控制 作者:Anoushka Alavilli*, Khai Nguyen*, Sam Schoedel*, Brian Plancher, Zachary Manchester Carnegie Mellon University, Barnard College 模型预测控制(Model-predictive control,MPC)是控制受复杂约束条件(complex constraints)影响的高

    2024年02月03日
    浏览(33)
  • LabVIEW开发微控制器控制的并行机器人的实时视觉图像处理

    LabVIEW开发微控制器控制的并行机器人的实时视觉图像处理         通过相机视觉,以对目标物体的不同颜色进行分类,并与平行机器人一起拾取和放置物体。通过使用MATLAB Simulink模拟合适的机器人工作空间来研究使用相机的效率和机器人的准确性。机械臂以使用运动学计算

    2024年02月09日
    浏览(38)
  • YASKAWA ERCR-NS01-B004-E1机器人控制器

    YASKAWA ERCR-NS01-B004-E1机器人控制器是一款由安川电机(Yaskawa Electric)生产的工业机器人控制器。该控制器主要用于控制和协调机器人的运动,包括机器人的轨迹规划、速度控制、姿态调整等。 ERCR-NS01-B004-E1控制器具有以下特点: 高性能:采用高速处理器和大容量存储器,支持

    2024年01月17日
    浏览(35)
  • FANUC机器人PRIO-621和PRIO-622设备和控制器没有运行故障处理

    如下图所示,新的机器人开机后提示报警: PRIO-621 设备没有运行 PRIO-622 控制器没有运行 我们首先查看下手册上的报警代码说明,如下图所示, 如下图所示,在PROFINET IO界面中,找到第4、5项: I/O控制装置和I/O设备均选择禁用 , 设置完成后,给控制柜断电再重新上电即可消

    2024年02月07日
    浏览(75)
  • ABB机器人将程序模块单个或批量导入控制器的具体方法和步骤

    在一个项目中,有可能会需要多个模块和程序文件,如果都是使用示教器来建立或编程,不是太方便且耗费时间较长。 通过RobotStudio可以单个/批量加载程序模块和参数。 具体的方法和步骤可参考以下内容 : 如下图所示,我们在电脑上创建一个离线的工作站项目,并创建了程

    2024年02月12日
    浏览(84)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包