我正在开发一个多人游戏的服务器,该游戏必须控制数千个在世界各地运行的生物。每个生物都有一个带有心跳方法的 AI,如果玩家在附近,每隔几毫秒/秒就会调用一次,这样他们就可以做出反应。
目前,人工智能使用枚举器作为“例程”,例如
IEnumerable WanderAround(int radius)
{
// Do something
}
它们从“状态方法”中调用,这些方法在 foreachs 中调用,在心跳中产生,因此您在每个滴答声中都回到同一个位置。
void OnHeartbeat()
{
// Do checks, maybe select a new state method...
// Then continue the current sequence
currentState.MoveNext();
}
自然也必须在循环中调用例程,因为否则它们不会执行。但由于我不是编写这些 AI 的人,而是不一定是程序员的新手,所以我在服务器启动之前预编译 AI(简单的 .cs 文件)。这给了我这样的 AI 脚本:
override IEnumerable Idle()
{
Do(WanderAround(400));
Do(Wait(3000));
}
override IEnumerable Aggro()
{
Do(Attack());
Do(Wait(3000));
}
替换Do
为迭代例程调用的 foreach 。
我真的很喜欢这种设计,因为 AI 易于理解,但功能强大。这不是简单的状态,但也不难理解/编写行为树。
现在到我实际的“问题”,我不喜欢Do
包装器,我不喜欢预编译我的脚本。但是我想不出没有循环的任何其他方法来实现它,由于冗长和要编写这些脚本的人的技能水平,我想隐藏它。
foreach(var r in Attack()) yield return r;
我希望有一种方法可以在没有显式循环的情况下调用例程,但这是不可能的,因为我必须从 state 方法中让步。
而且我不能使用 async/await,因为它不符合我所依赖的 tick 设计(AI 可能非常复杂,老实说,我不知道如何使用 async 来实现它)。另外,我只是交易Do()
,await
并没有太大的改进。
所以我的问题是:谁能想到摆脱那个循环包装器的方法?我愿意使用其他可以用作脚本的.NET语言(在服务器启动时编译它们),如果有一种以某种方式支持它的话。