BT3:库中基本类型——Tree和TreeNode

2023-11-11 08:30

本文主要是介绍BT3:库中基本类型——Tree和TreeNode,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

大家好,欢迎大家关注我的知乎专栏慢慢悠悠小马车


本文重点讲述BehaviorTree.CPP中的几个重要的基本的类定义,尤其是类所包含的数据。

Tree

定义在BehaviorTree.CPP/include/behaviortree_cpp_v3/bt_factory.h。有3个重要的public成员变量。

public:// 保存树的所有nodestd::vector<TreeNode::Ptr> nodes;// 保存所有blackboardstd::vector<Blackboard::Ptr> blackboard_stack;// 保存所有node注册信息std::unordered_map<std::string, TreeNodeManifest> manifests;

以BehaviorTree.CPP/examples/t06_subtree_port_remapping.cpp中的行为树为例,打印出以上3个容器的元素数量。

<root main_tree_to_execute = "MainTree"><BehaviorTree ID="MainTree"><Sequence name="main_sequence"><SetBlackboard output_key="move_goal" value="1;2;3" /><SubTree ID="MoveRobot" target="move_goal" output="move_result" /><SaySomething message="{move_result}"/></Sequence></BehaviorTree><BehaviorTree ID="MoveRobot"><Fallback name="move_robot_main"><SequenceStar><MoveBase       goal="{target}"/><SetBlackboard output_key="output" value="mission accomplished" /></SequenceStar><ForceFailure><SetBlackboard output_key="output" value="mission failed" /></ForceFailure></Fallback></BehaviorTree></root>

共有10个节点,2个blackboard(因为有2棵树),31个注册节点信息(可以理解为该tree认识31个节点,却只创建和包含了10个节点) 。

tree nodes count = 10
tree blackboard_stack count = 2
tree manifests count = 31

将节点的名称和注册节点信息的名称打印如下,可见一棵子树也会作为一个节点对待。而31个manifests中,就包含了BehaviorTree.CPP库提供的所有ControlNodes和DecoratorNodes,以及示例用的SaySomething和MoveBase。

tree root node = main_sequencenodes[1] = main_sequence
nodes[2] = SetBlackboard
nodes[3] = MoveRobot
nodes[4] = move_robot_main
nodes[5] = SequenceStar
nodes[6] = MoveBase
nodes[7] = SetBlackboard
nodes[8] = ForceFailure
nodes[9] = SetBlackboard
nodes[10] = SaySomething
manifests[1] = SaySomething
manifests[2] = Switch4
manifests[3] = Switch6
manifests[4] = BlackboardCheckDouble
manifests[5] = BlackboardCheckInt
manifests[6] = SubTree
manifests[7] = KeepRunningUntilFailure
manifests[8] = Switch5
manifests[9] = ReactiveSequence
manifests[10] = Parallel
manifests[11] = Delay
manifests[12] = SetBlackboard
manifests[13] = SequenceStar
manifests[14] = Fallback
manifests[15] = AlwaysSuccess
manifests[16] = ReactiveFallback
manifests[17] = Sequence
manifests[18] = Switch3
manifests[19] = Switch2
manifests[20] = AlwaysFailure
manifests[21] = IfThenElse
manifests[22] = WhileDoElse
manifests[23] = SubTreePlus
manifests[24] = ForceSuccess
manifests[25] = Inverter
manifests[26] = BlackboardCheckString
manifests[27] = RetryUntilSuccesful
manifests[28] = ForceFailure
manifests[29] = MoveBase
manifests[30] = Repeat
manifests[31] = Timeout

TreeNode

 定义在BehaviorTree.CPP/include/behaviortree_cpp_v3/tree_node.h。注意区分name_和registration_ID_的区别。

private:const std::string name_;  // 从xml获得的node名称,可没有,可重复NodeStatus status_;       // node的返回结果,即执行状态std::condition_variable state_condition_variable_;mutable std::mutex state_mutex_;StatusChangeSignal state_change_signal_;  // 订阅的信号const uint16_t uid_;      // 唯一IDNodeConfiguration config_;std::string registration_ID_; // 类型名称,一定与class name相同

以BehaviorTree.CPP/examples/t06_subtree_port_remapping.cpp中的行为树为例,打印出所有node的name和注册ID对比。可见registration_ID_和node的class name相同(非强制,下篇讲解),而name可以随意指定,如nodes[4],当不设置时默认是registration_ID_,如其中的nodes[2]和nodes[5]。

nodes[1].name=main_sequence, reg_id=Sequence
nodes[2].name=SetBlackboard, reg_id=SetBlackboard
nodes[3].name=MoveRobot, reg_id=SubTree
nodes[4].name=move_robot_main, reg_id=Fallback
nodes[5].name=SequenceStar, reg_id=SequenceStar
nodes[6].name=MoveBase, reg_id=MoveBase
nodes[7].name=SetBlackboard, reg_id=SetBlackboard
nodes[8].name=ForceFailure, reg_id=ForceFailure
nodes[9].name=SetBlackboard, reg_id=SetBlackboard
nodes[10].name=SaySomething, reg_id=SaySomething

NodeConfiguration中包含了blackboard的指针,和输入输出ports的映射信息。

typedef std::unordered_map<std::string, std::string> PortsRemapping;struct NodeConfiguration {Blackboard::Ptr blackboard;PortsRemapping input_ports;   // 输入port的映射关系PortsRemapping output_ports;  // 输出port的映射关系
};

打印SaySomething的input_ports如下,没有output ports。

input port: message --- {move_result}

 打印SetBlackboard的ports如下,该node比较特殊,output_key是个INOUT双向port,以后会单独介绍。

input port: output_key --- move_goal
input port: value --- 1;2;3output port: output_key --- move_goal

TreeNode类中提供了2个容易迷惑的接口,1个是虚函数executeTick(),1个是纯虚函数tick(),那么开发者应该实现和调用哪一个呢?

public:/// The method that should be used to invoke tick() and setStatus();virtual BT::NodeStatus executeTick();
protected:/// Method to be implemented by the uservirtual BT::NodeStatus tick() = 0;

executeTick()提供了默认实现,即先调用tick()然后设置返回的状态。而各种ControlNodes和DecoratorNodes,都是在tick()中调用child_node_->executeTick(),所以开发者只需实现子类的tick()就好了。

NodeStatus TreeNode::executeTick() {const NodeStatus status = tick();setStatus(status);return status;
}

行为树执行时会是这样的逻辑:

这篇关于BT3:库中基本类型——Tree和TreeNode的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/388788

相关文章

基本知识点

1、c++的输入加上ios::sync_with_stdio(false);  等价于 c的输入,读取速度会加快(但是在字符串的题里面和容易出现问题) 2、lower_bound()和upper_bound() iterator lower_bound( const key_type &key ): 返回一个迭代器,指向键值>= key的第一个元素。 iterator upper_bou

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【IPV6从入门到起飞】5-1 IPV6+Home Assistant(搭建基本环境)

【IPV6从入门到起飞】5-1 IPV6+Home Assistant #搭建基本环境 1 背景2 docker下载 hass3 创建容器4 浏览器访问 hass5 手机APP远程访问hass6 更多玩法 1 背景 既然电脑可以IPV6入站,手机流量可以访问IPV6网络的服务,为什么不在电脑搭建Home Assistant(hass),来控制你的设备呢?@智能家居 @万物互联

自定义类型:结构体(续)

目录 一. 结构体的内存对齐 1.1 为什么存在内存对齐? 1.2 修改默认对齐数 二. 结构体传参 三. 结构体实现位段 一. 结构体的内存对齐 在前面的文章里我们已经讲过一部分的内存对齐的知识,并举出了两个例子,我们再举出两个例子继续说明: struct S3{double a;int b;char c;};int mian(){printf("%zd\n",s

【编程底层思考】垃圾收集机制,GC算法,垃圾收集器类型概述

Java的垃圾收集(Garbage Collection,GC)机制是Java语言的一大特色,它负责自动管理内存的回收,释放不再使用的对象所占用的内存。以下是对Java垃圾收集机制的详细介绍: 一、垃圾收集机制概述: 对象存活判断:垃圾收集器定期检查堆内存中的对象,判断哪些对象是“垃圾”,即不再被任何引用链直接或间接引用的对象。内存回收:将判断为垃圾的对象占用的内存进行回收,以便重新使用。

flume系列之:查看flume系统日志、查看统计flume日志类型、查看flume日志

遍历指定目录下多个文件查找指定内容 服务器系统日志会记录flume相关日志 cat /var/log/messages |grep -i oom 查找系统日志中关于flume的指定日志 import osdef search_string_in_files(directory, search_string):count = 0

两个月冲刺软考——访问位与修改位的题型(淘汰哪一页);内聚的类型;关于码制的知识点;地址映射的相关内容

1.访问位与修改位的题型(淘汰哪一页) 访问位:为1时表示在内存期间被访问过,为0时表示未被访问;修改位:为1时表示该页面自从被装入内存后被修改过,为0时表示未修改过。 置换页面时,最先置换访问位和修改位为00的,其次是01(没被访问但被修改过)的,之后是10(被访问了但没被修改过),最后是11。 2.内聚的类型 功能内聚:完成一个单一功能,各个部分协同工作,缺一不可。 顺序内聚:

Mysql BLOB类型介绍

BLOB类型的字段用于存储二进制数据 在MySQL中,BLOB类型,包括:TinyBlob、Blob、MediumBlob、LongBlob,这几个类型之间的唯一区别是在存储的大小不同。 TinyBlob 最大 255 Blob 最大 65K MediumBlob 最大 16M LongBlob 最大 4G

C 语言的基本数据类型

C 语言的基本数据类型 注:本文面向 C 语言初学者,如果你是熟手,那就不用看了。 有人问我,char、short、int、long、float、double 等这些关键字到底是什么意思,如果说他们是数据类型的话,那么为啥有这么多数据类型呢? 如果写了一句: int a; 那么执行的时候在内存中会有什么变化呢? 橡皮泥大家都玩过吧,一般你买橡皮泥的时候,店家会赠送一些模板。 上

Oracle type (自定义类型的使用)

oracle - type   type定义: oracle中自定义数据类型 oracle中有基本的数据类型,如number,varchar2,date,numeric,float....但有时候我们需要特殊的格式, 如将name定义为(firstname,lastname)的形式,我们想把这个作为一个表的一列看待,这时候就要我们自己定义一个数据类型 格式 :create or repla