状态机和行为树的搭配使用

2024-06-20 12:52
文章标签 使用 行为 搭配 状态机

本文主要是介绍状态机和行为树的搭配使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

下面是一个完整的TypeScript代码示例,展示了如何将状态机和行为树结合起来使用。状态机控制智能体的状态,行为树负责智能体的智能决策。在执行决策前,行为树会检查状态机的当前状态。

// 定义状态枚举
enum BodyState {IDLE,MOVING,ATTACKING
}// 状态机类,控制智能体的状态
class StateMachine {private state: BodyState;constructor() {this.state = BodyState.IDLE;}getState(): BodyState {return this.state;}setState(state: BodyState): void {this.state = state;}
}// 智能体类,拥有一个状态机
class Entity {private stateMachine: StateMachine;constructor() {this.stateMachine = new StateMachine();}getStateMachine(): StateMachine {return this.stateMachine;}getState(): BodyState {return this.stateMachine.getState();}// 智能体可以拥有更多的属性和方法
}// 行为树的相关代码
enum Status {SUCCESS,FAILURE,RUNNING
}abstract class Node {abstract execute(entity: Entity): Status;
}abstract class CompositeNode extends Node {protected children: Node[] = [];addChild(child: Node): void {this.children.push(child);}
}class SelectorNode extends CompositeNode {execute(entity: Entity): Status {for (const child of this.children) {const status = child.execute(entity);if (status === Status.SUCCESS) {return Status.SUCCESS;}}return Status.FAILURE;}
}class SequenceNode extends CompositeNode {execute(entity: Entity): Status {for (const child of this.children) {const status = child.execute(entity);if (status === Status.FAILURE) {return Status.FAILURE;}}return Status.SUCCESS;}
}abstract class ConditionNode extends Node {abstract check(entity: Entity): boolean;execute(entity: Entity): Status {return this.check(entity) ? Status.SUCCESS : Status.FAILURE;}
}abstract class ActionNode extends Node {abstract performAction(entity: Entity): boolean;execute(entity: Entity): Status {return this.performAction(entity) ? Status.SUCCESS : Status.FAILURE;}
}abstract class DecoratorNode extends Node {protected child: Node;constructor(child: Node) {super();this.child = child;}
}class InverterNode extends DecoratorNode {execute(entity: Entity): Status {const status = this.child.execute(entity);if (status === Status.SUCCESS) {return Status.FAILURE;} else if (status === Status.FAILURE) {return Status.SUCCESS;} else {return status;}}
}class RepeaterNode extends DecoratorNode {private maxRepeats: number;private stopOnSuccess: boolean;private stopOnFailure: boolean;constructor(child: Node, maxRepeats: number, stopOnSuccess: boolean = false, stopOnFailure: boolean = false) {super(child);this.maxRepeats = maxRepeats;this.stopOnSuccess = stopOnSuccess;this.stopOnFailure = stopOnFailure;}execute(entity: Entity): Status {let count = 0;while (this.maxRepeats === -1 || count < this.maxRepeats) {const status = this.child.execute(entity);if (this.stopOnSuccess && status === Status.SUCCESS) {return Status.SUCCESS;}if (this.stopOnFailure && status === Status.FAILURE) {return Status.FAILURE;}count++;}return Status.SUCCESS;}
}// 定义行为树节点
class IsEnemyInRange extends ConditionNode {check(entity: Entity): boolean {// 判断敌人是否在范围内return Math.random() < 0.5; // 50% 概率敌人在范围内}
}class ChaseEnemy extends ActionNode {performAction(entity: Entity): boolean {const stateMachine = entity.getStateMachine();if (stateMachine.getState() === BodyState.IDLE || stateMachine.getState() === BodyState.MOVING) {console.log("Chasing the enemy!");stateMachine.setState(BodyState.MOVING);return true; // 假设追击成功}return false;}
}class AttackEnemy extends ActionNode {performAction(entity: Entity): boolean {const stateMachine = entity.getStateMachine();if (stateMachine.getState() === BodyState.MOVING) {console.log("Attacking the enemy!");stateMachine.setState(BodyState.ATTACKING);return true; // 假设攻击成功}return false;}
}// 创建智能体实例
const entity = new Entity();// 创建行为树
const root = new SelectorNode();const sequence = new SequenceNode();
root.addChild(sequence);const condition = new IsEnemyInRange();
sequence.addChild(condition);const chase = new ChaseEnemy();
const repeater = new RepeaterNode(chase, 3); // 重复执行ChaseEnemy 3次
sequence.addChild(repeater);const attack = new AttackEnemy();
sequence.addChild(attack);// 执行为树
const status = root.execute(entity);
console.log(`Behavior tree execution status: ${Status[status]}`);
console.log(`Current body state: ${BodyState[entity.getState()]}`);

在这个示例中,智能体类拥有一个状态机,通过行为树来控制智能体的状态。行为树会根据状态机的当前状态来决策执行某个行为,同时更新状态机的状态。 

这篇关于状态机和行为树的搭配使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C语言中联合体union的使用

本文编辑整理自: http://bbs.chinaunix.net/forum.php?mod=viewthread&tid=179471 一、前言 “联合体”(union)与“结构体”(struct)有一些相似之处。但两者有本质上的不同。在结构体中,各成员有各自的内存空间, 一个结构变量的总长度是各成员长度之和。而在“联合”中,各成员共享一段内存空间, 一个联合变量

Tolua使用笔记(上)

目录   1.准备工作 2.运行例子 01.HelloWorld:在C#中,创建和销毁Lua虚拟机 和 简单调用。 02.ScriptsFromFile:在C#中,对一个lua文件的执行调用 03.CallLuaFunction:在C#中,对lua函数的操作 04.AccessingLuaVariables:在C#中,对lua变量的操作 05.LuaCoroutine:在Lua中,

Vim使用基础篇

本文内容大部分来自 vimtutor,自带的教程的总结。在终端输入vimtutor 即可进入教程。 先总结一下,然后再分别介绍正常模式,插入模式,和可视模式三种模式下的命令。 目录 看完以后的汇总 1.正常模式(Normal模式) 1.移动光标 2.删除 3.【:】输入符 4.撤销 5.替换 6.重复命令【. ; ,】 7.复制粘贴 8.缩进 2.插入模式 INSERT

Lipowerline5.0 雷达电力应用软件下载使用

1.配网数据处理分析 针对配网线路点云数据,优化了分类算法,支持杆塔、导线、交跨线、建筑物、地面点和其他线路的自动分类;一键生成危险点报告和交跨报告;还能生成点云数据采集航线和自主巡检航线。 获取软件安装包联系邮箱:2895356150@qq.com,资源源于网络,本介绍用于学习使用,如有侵权请您联系删除! 2.新增快速版,简洁易上手 支持快速版和专业版切换使用,快速版界面简洁,保留主

如何免费的去使用connectedpapers?

免费使用connectedpapers 1. 打开谷歌浏览器2. 按住ctrl+shift+N,进入无痕模式3. 不需要登录(也就是访客模式)4. 两次用完,关闭无痕模式(继续重复步骤 2 - 4) 1. 打开谷歌浏览器 2. 按住ctrl+shift+N,进入无痕模式 输入网址:https://www.connectedpapers.com/ 3. 不需要登录(也就是

Toolbar+DrawerLayout使用详情结合网络各大神

最近也想搞下toolbar+drawerlayout的使用。结合网络上各大神的杰作,我把大部分的内容效果都完成了遍。现在记录下各个功能效果的实现以及一些细节注意点。 这图弹出两个菜单内容都是仿QQ界面的选项。左边一个是drawerlayout的弹窗。右边是toolbar的popup弹窗。 开始实现步骤详情: 1.创建toolbar布局跟drawerlayout布局 <?xml vers

C#中,decimal类型使用

在Microsoft SQL Server中numeric类型,在C#中使用的时候,需要用decimal类型与其对应,不能使用int等类型。 SQL:numeric C#:decimal

探索Elastic Search:强大的开源搜索引擎,详解及使用

🎬 鸽芷咕:个人主页  🔥 个人专栏: 《C++干货基地》《粉丝福利》 ⛺️生活的理想,就是为了理想的生活! 引入 全文搜索属于最常见的需求,开源的 Elasticsearch (以下简称 Elastic)是目前全文搜索引擎的首选,相信大家多多少少的都听说过它。它可以快速地储存、搜索和分析海量数据。就连维基百科、Stack Overflow、

flask 中使用 装饰器

因为要完成毕业设计,我用到fountain code做数据恢复。 于是在github上下载了fountain code的python原代码。 github上的作者用flask做了fountain code的demo。 flask是面向python的一个网站框架。 里面有用到装饰器。 今天笔试的时候,我也被问到了python的装饰器。

mathematica的使用

因为做实验用到Bloom filter这一技术,Bloom filter里面的数学公式可以用来画图。 那么用什么画图软件比较好呢? 当然是Mathematica啦。 利用代码Plot[{y=x},{x,0,100}] 就可以画出比较好的图 简直nice