ROS2 的行为树 — 第 1 部分:解锁高级机器人决策和控制

一、说明

        在复杂而迷人的机器人世界中,行为树(BT)已成为决策过程中不可或缺的一部分。它们提供了一种结构化模块化高效的方法来对机器人的行为进行编程。BT起源于视频游戏行业,用于控制非玩家角色,他们在机器人领域找到了归宿,他们擅长管理无数的任务和条件。

        用于机器人导航的Nav2和操作框架MoveIt等机器人软件使用行为树来处理复杂的动作和条件。

        在这篇文章中,我们将探索行为树以彻底理解它们。我们将首先分解行为树的基本部分,并清楚地了解它们的工作原理。然后,我们将进入实际领域,编写代码来创建和使用行为树,独立于任何特定的平台或框架。

但我们不会止步于此。一旦我们掌握了基础知识,我们将更进一步,将我们的行为树与ROS2(机器人操作系统2)集成,ROS<>是一个用于编写机器人软件的灵活框架。这将展示如何在现实世界的机器人环境中使用行为树,增强我们自主系统的决策能力。

二、行为树的基本概念 

        将行为树 (BT) 视为视频游戏角色、机器人或任何其他需要决定下一步操作的自主代理的决策图。

        想象一下,你正在导演一部戏剧,但你的演员是机器人。作为导演,你需要一种方法来指导你的演员,告诉他们该做什么以及什么时候做。行为树(BT)就像你的脚本,详细说明你的机器人演员需要做出的每一个动作。就像在真实的戏剧中一样,当导演叫“动作”(或者,在这种情况下,“滴答”)时,节目(或任务)就开始了。

        那么,什么是“刻度”?在BT的上下文中,tick是从树的根节点(或起点)到其子节点的信号,敦促它们执行任务。节点仅在收到即时报价时“采取行动”。

        现在,这个系统的美妙之处在于它的反馈机制。节点开始执行其任务后,它会使用以下三种状态之一与控制器(其父节点)通信其进度:正在运行成功失败

正在运行:这种状态有点像在说,“我在上面,但我还没有完成。
成功:这相当于,“我已经尽了自己的一份力量,而且进展顺利。
失败:这就像说,“我试过了,但我做不到。

        根据这些状态,导演决定将下一个即时报价发送到何处。这是一个持续到任务完成或不再有效的循环。

        行为树由两种主要类型的节点组成:控制流节点和执行节点。让我们更深入地了解它们。

        控制流节点就像导演的助手,他们帮助决定演员下一步应该做什么。主要有四种类型:

        序列(→)节点:这就像一个严格的助手,他要求每个任务都必须完美地完成,顺序,然后才能继续下一个任务。如果任何任务失败,序列将停止并报告失败。

        回退 (?节点(也称为选择器):这是一个更灵活的助手,他有一个任务列表并逐个尝试,直到一个成功。如果所有任务都失败,则回退报告失败。

        并行 (⇒)节点: 这是一个可以同时处理多项任务的助手,指导多个演员同时执行他们的任务。并行节点根据一组必须成功的任务来定义其成功。

        装饰器节点:这个助手有点特别,因为它只处理一个演员,但可以通过各种方式修改演员的行为。有不同种类的装饰器,每种装饰器都为任务提供了独特的转折。

执行节点是参与者,即执行任务(操作)或评估某些条件(条件)的参与者。

        操作节点: 当此参与者收到勾号时,它会启动其任务,使控制器保持“正在运行”状态的更新,最后报告“成功”或“失败”。

        条件节点: 这个演员更像是一个抽查者。它评估是否满足某些条件,如果条件为真,则立即报告“成功”,如果条件为假,则立即报告“失败”。

        现在您已经了解了基础知识,您可以使用行为树指导您的机器人游戏。请记住,此框架旨在提高灵活性和适应性,使您的机器人参与者能够快速响应变化并执行复杂、细致入微的任务。

三、BT 的节点类型

        让我们考虑一个机器人手臂上下文中的行为树,它可以捡起球并将其放置在球门位置。

高级BT执行一项任务,包括首先找到,然后挑选并最终放置球

  1. BT 以序列节点作为主控制器开始。该节点对机器人有两个主要任务:“查找球”和“放置球”。
  2. 第一个任务“找球”涉及一个机器人,该机器人被分配来定位球。
  3. 第二个任务“放置一个球”更为复杂。此任务由另一个“序列”节点管理,该节点可确保按顺序执行一系列操作。此序列涉及任务“挑选球”。
  4. “拾球”任务由两个回退节点管理。这些节点中的每一个都有不同的方法来引导机器人。
  • 第一个回退节点确保机器人检查“球是否接近”,如果是,则指示机器人“接近球”。
  • 第二个回退节点负责检查“球是否被抓住”,如果没有,则指示机器人“抓住球”。

        请记住,在BT中,节点仅在收到信号或“滴答”时才起作用。一旦任务启动,机器人就会将其状态传达回其父节点。循环一直持续到所有任务完成或无法完成。

        到目前为止,我们已经详细讨论了行为树的概念和机制。现在,是时候将这些想法付诸行动了。

        在深入研究实际编码之前,我们需要设置行为树包。在以下各节中,我们将引导您完成设置此包的过程,之后我们将探索将行为树变为现实的代码。

        要在 Ubuntu 20.04 上设置并运行我们的第一个 BehaviorTree.CPP 示例,请执行以下步骤:

# Install the required dependencies:
# Open a terminal and run the following commands to install the necessary 
# dependencies:
sudo apt-get update
sudo apt-get install -y libzmq3-dev libboost-dev libboost-system-dev libboost-filesystem-dev libboost-thread-dev libprotobuf-dev protobuf-compiler libmsgsl-dev libgtest-dev cmake# Build and install BehaviorTree.CPP:
# Clone the BehaviorTree.CPP repository, build, and install the library:
git clone https://github.com/BehaviorTree/BehaviorTree.CPP.git
cd BehaviorTree.CPP
mkdir build
cd build
cmake ..
make -j$(nproc)
sudo make install

让我们从使用 BehaviorTree 的简单示例开始.CPP将帮助您了解基础知识并熟悉语法。

mkdir ~/example1
cd ~/example1
touch main.cpp && touch CMakeLists.txt
// main.cpp#include <behaviortree_cpp/bt_factory.h>
#include <behaviortree_cpp/behavior_tree.h>using namespace BT;class SaySomething : public SyncActionNode
{
public:SaySomething(const std::string& name, const NodeConfiguration& config): SyncActionNode(name, config){}static PortsList providedPorts(){return { InputPort<std::string>("message") };}NodeStatus tick() override{std::string message;getInput("message", message);std::cout << "Robot says: " << message << std::endl;return NodeStatus::SUCCESS;}
};static const char* xml_text = R"(
<root><BehaviorTree><SaySomething message="Hello, World!" /></BehaviorTree>
</root>
)";int main()
{BehaviorTreeFactory factory;factory.registerNodeType<SaySomething>("SaySomething");auto tree = factory.createTreeFromText(xml_text);tree.tickWhileRunning();return 0;
}
# CMakeLists.txt
cmake_minimum_required(VERSION 3.5)
project(bt_example)set(CMAKE_CXX_STANDARD 17)find_package(behaviortree_cpp REQUIRED)add_executable(bt_example main.cpp)
target_link_libraries(bt_example BT::behaviortree_cpp)
# Now, build and run the example:
mkdir build
cd build
cmake ..
make
./bt_example

        太棒了,您已经获得了第一个BehaviorTree.CPP 代码片段!该脚本代表了一个基本的行为树,其中机器人(您的演员)有一个简单的任务:说些什么。让我们来分解一下:

        该类扩展了该类,该类是 BehaviorTree.CPP 提供的执行节点类型之一。这是一个同步操作节点,这意味着它不返回“正在运行”状态,而只返回“成功”或“失败”。在我们的例子中,它将始终返回“成功”。SaySomethingSyncActionNodeSyncActionNode

        该方法是在勾选此节点时调用的主要方法。在这里,它检索输入消息并将其输出到控制台。由于它每次都成功执行其任务,因此它会返回 .tick()NodeStatus::SUCCESS

        定义为 的 XML 文本表示行为树的结构。它包含一个节点,带有消息“Hello, World!xml_textSaySomething

        最后,该函数创建一个 ,注册节点类型,从 XML 文本创建树,并在树运行时勾选树。这让机器人说“你好,世界!mainBehaviorTreeFactorySaySomething

        现在,有了基础知识,您就可以转向更复杂的行为树,例如拾取和放置树。在选取和放置树中,您将引入更多控制流节点和执行节点,将它们交织在一起以实现所需的行为。此处提供了此示例的完整代码。

#include <iostream>#include <behaviortree_cpp/bt_factory.h>
#include <behaviortree_cpp/behavior_tree.h>using namespace BT;// Define FindBall action node
class FindBall : public BT::SyncActionNode 
{public:FindBall(const std::string& name) : BT::SyncActionNode(name, {}){}// Define the tick() function for the FindBall nodeNodeStatus tick() override{std::cout << "[⚓ FindBall ] => \t" << this->name() << std::endl; return BT::NodeStatus::SUCCESS;}
};// Define PickBall action node
class PickBall : public BT::SyncActionNode
{public:PickBall(const std::string& name) : BT::SyncActionNode(name, {}){}// Define the tick() function for the PickBall nodeNodeStatus tick() override{std::cout << "[⚓ PickBall ] => \t" << this->name() << std::endl;return BT::NodeStatus::SUCCESS;}
};// Define PlaceBall action node
class PlaceBall : public BT::SyncActionNode
{public:PlaceBall(const std::string& name) : BT::SyncActionNode(name, {}){}// Define the tick() function for the PlaceBall nodeNodeStatus tick() override{std::cout << "[⚓ PlaceBall ] => \t" << this->name() << std::endl;return BT::NodeStatus::SUCCESS;}
};// Define GripperInterface class for interacting with the gripper
class GripperInterface
{private:bool _opened;public:GripperInterface() : _opened(true){}NodeStatus open(){_opened = true;std::cout << "GripperInterface::open" << std::endl;return BT::NodeStatus::SUCCESS;}NodeStatus close(){std::cout << "GripperInterface::close" << std::endl;_opened = false;return BT::NodeStatus::SUCCESS;}
};// Define BallClose condition function
BT::NodeStatus BallClose()
{std::cout << "[ Close to ball: NO ]" << std::endl;return BT::NodeStatus::FAILURE;
}// Define BallGrasped condition function
BT::NodeStatus BallGrasped()
{std::cout << "[ Grasped: NO ]" << std::endl;return BT::NodeStatus::FAILURE;
}// Define the behavior tree structure in XML format
static const char* xml_text = R"(<root main_tree_to_execute = "MainTree" ><BehaviorTree ID="MainTree"><Sequence name="root_sequence"><FindBall   name="found_ok"/><Sequence><Fallback><BallClose   name="no_ball"/><PickBall    name="approach_ball"/></Fallback><Fallback><BallGrasped   name="no_grasp"/><GraspBall  name="grasp_ball"/></Fallback></Sequence><PlaceBall   name="ball_placed"/></Sequence></BehaviorTree></root>)";int main()
{    BehaviorTreeFactory factory;// Register custom action nodes with the factoryfactory.registerNodeType<FindBall>("FindBall");factory.registerNodeType<PickBall>("PickBall");factory.registerNodeType<PlaceBall>("PlaceBall");// Register custom condition nodes with the factoryfactory.registerSimpleCondition("BallClose", std::bind(BallClose));factory.registerSimpleCondition("BallGrasped", std::bind(BallGrasped));// Create an instance of GripperInterfaceGripperInterface gripper;// Register a simple action node using the GripperInterface instancefactory.registerSimpleAction("GraspBall", std::bind(&GripperInterface::close, &gripper));// Create the behavior tree using the XML descriptionauto tree = factory.createTreeFromText(xml_text);// Run the behavior tree until it finishestree.tickWhileRunning();return 0;
}
# Output
[⚓ FindBall ] =>       found_ok
[ Close to ball: NO ]
[⚓ PickBall ] =>       approach_ball
[ Grasped: NO ]
GripperInterface::close
[⚓ PlaceBall ] =>      cube_placed

        太好了,您创建了一个更复杂的行为树!此脚本表示机器人的拾取和放置操作,由查找、拾取和放置球组成。让我们剖析一下:

  1. 您已经创建了三个操作节点:、 和 。这些类中的每一个都扩展并实现函数,该函数记录操作的名称并返回 .FindBallPickBallPlaceBallSyncActionNodetick()NodeStatus::SUCCESS
  2. 您已经定义了一个类,该类表示机器人的抓手。它可以打开和关闭夹持器,两个操作返回。GripperInterfaceNodeStatus::SUCCESS
  3. 您已经定义了两个条件函数:和 。这些功能表示机器人在捡球之前需要进行的检查。他们目前总是返回,模拟球没有接近并且机器人还没有抓住球的场景。BallCloseBallGraspedNodeStatus::FAILURE
  4. XML 字符串定义行为树。它从一个序列节点开始,该节点首先尝试找到球 ()。找到球后,它会尝试拾取球,这是由两个回退节点组成的另一个序列节点。第一个回退节点检查球是否接近 (),如果没有,机器人将接近球 ()。第二个回退节点检查球是否被抓住(),如果没有,机器人会抓住球()。一旦球被捡起,机器人就会放置球()。xml_textFindBallBallClosePickBallBallGraspedGraspBallPlaceBall
  5. 最后,该函数注册自定义操作和条件节点,创建 的实例,使用 注册操作,从 XML 文本创建树,并在树运行时勾选树。mainGripperInterfaceGraspBallGripperInterface

        此代码提供了机器人如何使用行为树执行一系列任务的实际演示,并在此过程中进行检查。但是,请注意,当前条件始终返回 。为了使此示例更加逼真,您可能需要根据机器人的实际状态及其环境更新这些条件。NodeStatus::FAILURE

        这是这篇文章的总结!您已经了解了行为树、其结构和组件,甚至使用拾取和放置示例实现了复杂的树。

        请继续关注我们的下一篇文章,我们将把事情提升一个档次。我们将介绍 ROS2(机器人操作系统 2)与行为树的集成,为您的机器人应用程序创建复杂、健壮和自适应行为开辟更多可能性。下一篇文章见!

        我希望这些信息对您有所帮助,并且您发现它很有用。如果您有任何问题或意见,请随时告诉我。感谢您的反馈,并很乐意帮助回答您可能遇到的任何问题。感谢您抽出宝贵时间阅读本文。

        我总是在LinkedIn上分享有趣的文章和更新,所以如果您想随时了解情况,请随时在平台上关注我。杰加特桑·尚穆甘

参考资料:
1.行为树官方文档
2.机器人和人工智能中的行为树:简介
3.导航2行为树

Nav2 Behavior Trees — Nav2 1.0.0 documentation (ros.org) 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/141150.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

FPGA板卡启动以及LED灯带调试

环境配置 软件&#xff1a; MobaXterm&#xff08;free版本即可&#xff09;下载教程参考&#xff1a; MobaXterm&#xff08;终端工具&#xff09;下载&安装&使用教程_蜗牛也不慢......的博客-CSDN博客 Win32 Disklmager 下载教程参考&#xff1a; 不分类工具&am…

公众号迁移多久可以完成?

公众号账号迁移的作用是什么&#xff1f;只能变更主体吗&#xff1f;长期以来&#xff0c;由于部分公众号在注册时&#xff0c;主体不准确的历史原因&#xff0c;或者公众号主体发生合并、分立或业务调整等现实状况&#xff0c;在公众号登记主体不能对应实际运营人的情况下&…

在北京多有钱能称为富

背景 首先声明&#xff0c;此讨论仅限个人的观点&#xff0c;因为我本身不富嘛&#xff0c;所以想法应该非常局限。 举个栗子 富二代问我朋友&#xff0c;100~1000w之间&#xff0c;推荐一款车&#xff1f; 一开始听到这个问题的时候&#xff0c;有被唬住&#xff0c;觉得预…

基于Python+Django的热门旅游景点数据分析系统的设计与实现(源码+lw+部署文档+讲解等)

前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作者&#xff0c;博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战✌&#x1f497; &#x1f447;&#x1f3fb;…

什么是IoT数字孪生?

数字孪生是资产或系统的实时虚拟模型&#xff0c;它使用来自连接的物联网传感器的数据来创建数字表示。数字孪生允许您从任何地方实时监控设备、资产或流程。数字孪生用于多种目的&#xff0c;例如分析性能、监控问题或在实施之前运行测试。从物联网数字孪生中获得的见解使用户…

18795-2012 茶叶标准样品制备技术条件

声明 本文是学习GB-T 18795-2012 茶叶标准样品制备技术条件. 而整理的学习笔记,分享出来希望更多人受益,如果存在侵权请及时联系我们 1 范围 本标准规定了各类茶叶(除再加工茶)标准样品的制备、包装、标签、标识、证书和有效期。 本标准适用于各类茶叶(除再加工茶)感官品质…

怎么在OPPO手机桌面上添加文字?便签桌面插件添加教程

很多年轻女性在选择手机时&#xff0c;都比较青睐于设计时尚靓丽、轻薄且续航好、系统流畅、拍照清晰的OPPO手机&#xff0c;并且OPPO为不同的用户提供了高中低不同价格档位的手机型号&#xff0c;能够满足绝大多数女性消费者的使用需求。 不过有不少OPPO手机用户表示&#xf…

华为NFC设置教程(门禁卡/公交卡/校园卡等)

今天把华为NFC设置教程分享给大家 出门带门禁卡、校园卡、银行卡、身份证……东西又多&#xff0c;携带又麻烦&#xff0c;还容易搞丢&#xff0c;有没有一种方法可以把它们都装下&#xff1f;有&#xff01;只要一部手机&#xff0c;出门不带卡包&#xff0c;各种证件&#x…

【知识分享】Java获取全年每个月的有几周且每周是几号到几号

加哥本周给大家分享一期怎么用java把全年每个月有几周&#xff0c;本周是几号到几号的工具类。便于大家根据需求获取想要的形式进行改造。话不多说&#xff0c;直接给大家上代码。 package com.techfantasy.common.utils; import com.techfantasy.common.entity.DateRange; i…

安全测试之w3af 安装

背景&#xff1a; 学习安全测试时&#xff0c;离不开一些安全扫描工具&#xff0c;在类目众多的工具中&#xff0c;w3af是个绕不开的集成工具。由于没有linux环境&#xff0c;故在windows下利用wsl进行部署。亦可通过其它虚拟机安装亦可。 借鉴&#xff1a;Win10下安装w3af_Da…

NSSCTF做题

[第五空间 2021]WebFTP 打开题目 发现是登录的界面 用admin和password试一下发现不行 用dirsearch扫一下 发现了git泄露 但是用githack下载不下来文件 去网上查了一下webftp 发现是一个在线php文件管理系统 WebFTP——在线FTP工具:强大的PHP在线文件管理系统-時日 在这篇博客…

leetcode top100(20) 搜索二维矩阵 II

编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target 。该矩阵具有以下特性&#xff1a; 每行的元素从左到右升序排列。每列的元素从上到下升序排列。 示例 1&#xff1a; 输入&#xff1a;matrix [[1,4,7,11,15],[2,5,8,12,19],[3,6,9,16,22],[10,13,14,17,2…

6.1 使用scikit-learn构建模型

6.1 使用scikit-learn构建模型 6.1.1 使用sklearn转换器处理数据6.1.2 将数据集划分为训练集和测试集6.1.3 使用sklearn转换器进行数据预处理与降维1、数据预处理2、PCA降维算法 代码 scikit-learn&#xff08;简称sklearn&#xff09;库整合了多种机器学习算法&#xff0c;可以…

mysql Your password does not satisfy the current policy requirements

在修改密码时遇到 Your password does not satisfy the current policy requirements 原因&#xff1a;您的密码不符合当前策略要求&#xff0c;最好是把密码设置成复杂的&#xff0c;包括字母大小写、数字、特殊字符。 如果你还是先把数据库密码改简单&#xff0c;比如你本地…

C++【个人笔记1】

1.C的初识 1.1 简单入门 #include<iostream> using namespace std; int main() {cout << "hello world" << endl;return 0; } #include<iostream>; 预编译指令&#xff0c;引入头文件iostream.using namespace std; 使用标准命名空间cout …

AUTOSAR 面试知识回顾

如果答不上来&#xff0c;就讲当时做了什么 1. Ethernet基础: 硬件接口&#xff1a; ECU到PHY&#xff1a; data 是MII总线&#xff0c; 寄存器控制是SMI总线【MDCMDIO两根线, half duplex】PHY输出(100BASE-T1)&#xff1a; MDI总线&#xff0c;2 wire 【T1: twisted 1 pair …

【GO】LGTM_Grafana_gozero_配置trace(4)_代码实操及追踪

最近在尝试用 LGTM 来实现 Go 微服务的可观测性&#xff0c;就顺便整理一下文档。 Tempo 会分为 4 篇文章&#xff1a; Tempo 的架构官网测试实操跑通gin 框架发送 trace 数据到 tempogo-zero 微服务框架发送数据到 tempo 本文就是写一下如何在 go-zero 微服务框架里面配置 t…

golang入门笔记——pprof性能分析

文章目录 简介runtime/pprof的使用命令行交互网络服务性能分析pprof与性能测试结合压测工具go-wrk 简介 golang性能分析工具pprof的8个指标 1.性能分析的5个方面&#xff1a;CPU、内存、I/O、goroutine&#xff08;协程使用情况和泄漏检查&#xff09;、死锁检测以及数据竟态…

一篇文章成为递归大神:MySQL递归查询(with recursive)

理论原理 1、MySQL with Recursive是什么&#xff1f; MySQL with Recursive是一种基于递归思想的MySQL查询方式&#xff0c;可以实现对数据的递归查询和处理&#xff0c;返回符合条件的数据。在MySQL 8.0版本中&#xff0c;该功能被正式引入。 2、MySQL with Recursive有什么…

【kubernetes】使用virtual-kubelet扩展k8s

1 何为virtual-kubelet&#xff1f; kubelet是k8s的agent&#xff0c;负责监听Pod的调度情况&#xff0c;并运行Pod。而virtual-kubelet不是真实跑在宿主机上的&#xff0c;而是一个可以跑在任何地方的进程&#xff0c;该进程向k8s伪装成一个真实的Node&#xff0c;但是实际的…