转向行为! steering behaviors !

白玉无冰

共 7781字,需浏览 16分钟

 ·

2021-05-29 14:12

寻找 seek 避开 flee 到达 arrive 追逐 purse 躲避 evade 群落 flock !

效果

转向行为旨在通过使用简单的力来帮助自主角色以逼真的方式运动,这些简单的力结合起来可以围绕角色的环境产生逼真的,即兴的导航。它们不是基于涉及路径规划或全局计算的复杂策略,而是使用本地信息,例如邻居的部队。这使它们易于理解和实施,但仍然能够产生非常复杂的运动模式。

群落效果:

追逐偏移:

视频预览[欢迎关注,获得最新研究成果]:

实现

实现转向行为抓住以下两个点:

  • 转向力
  • 周围角色对象

角色类需要包含以下几个基本属性:

  • 位置
  • 速度
  • 转向力

每帧根据合力与加速度,速度与位置的关系,更新位置。

this._steeredForce.multiplyScalar(1 / this.mass)
this.velocity.add(this._steeredForce)
this._steeredForce.set(00)
this._position.add(this.velocity)

每种行为,都会算出有个期望速度,根据这个期望速度算出一个转向力加上去即可。

寻找 seek 与 避开 flee

寻找的期望速度直接指向目标点,转向力是期望速度减去当前速度。避开正好相反。

seek(target: Vec2): void {
    const desiredVelocity: Vec2 = Vec2.subtract(temp_v2, target, this._position).normalize()
    desiredVelocity.multiplyScalar(this.maxSpeed);
    const force: Vec2 = desiredVelocity.subtract(this.velocity);
    this._steeredForce.add(force);
}

flee(target: Vec2): void {
    const desiredVelocity: Vec2 = Vec2.subtract(temp_v2, target, this._position).normalize()
    desiredVelocity.multiplyScalar(this.maxSpeed);
    const force: Vec2 = desiredVelocity.subtract(this.velocity);
    this._steeredForce.subtract(force);
}

到达 arrive

与寻找的期望速度类似,方向是一样的,区别是快到的时候速度会减少。

可以加一个距离控制,在还没达到这个距离时,先以最快的速度过去,比较近的时候减缓速度。

arrive(target: Vec2): void {
    const desiredVelocity: Vec2 = Vec2.subtract(temp_v2, target, this._position).normalize()
    const dist: number = Vec2.distance(this._position, target)
    if (dist > this.arrivalThreshold) {
        desiredVelocity.multiplyScalar(this.maxSpeed);
    } else {
        desiredVelocity.multiplyScalar(this.maxSpeed * dist / this.arrivalThreshold);
    }
    const force: Vec2 = desiredVelocity.subtract(this.velocity);
    this._steeredForce.add(force);
}

追逐 purse 与 躲避 evade

追逐与躲避会预测目标将会移动到的位置,最后再调用寻找和避开。

其中,追逐时,如果刚好在正前方,就不用预测目标移动的位置,直接飞过去就好。

pursue(target: Vehicle): void {
    const toTarget = Vec2.subtract(temp_v2, target.position, this._position)
    if (toTarget.dot(this.heading) > 0 && this.heading.dot(target.heading) < -0.95) {
        // 如果面对面,正好在前面,就直接飞过去
        this.seek(target.position)
    } else {
        const lookAheadTime: number = Vec2.distance(this._position, target.position) / (this.maxSpeed + target.velocity.length());
        const predictedTarget: Vec2 = Vec2.add(temp2_v2, target.position, Vec2.multiplyScalar(temp_v2, target.velocity, lookAheadTime));
        this.seek(predictedTarget)
    }
}

evade(target: Vehicle): void {
    const lookAheadTime: number = Vec2.distance(this._position, target.position) / (this.maxSpeed + target.velocity.length())
    const predictedTarget: Vec2 = Vec2.add(temp2_v2, target.position, Vec2.multiplyScalar(temp_v2, target.velocity, lookAheadTime));
    this.flee(predictedTarget)
}    

追逐偏移 pursueOffset

追逐偏移使得角色之间保持指定位置的偏移。

先初始算出偏移位置,再算偏移预期位置,最后调用到达。

pursueOffset(target: Vehicle, offset: Vec2): void {
    const localOffset = temp_v2.set(
        target.side.x * offset.x + target.side.y * offset.y,
        target.heading.x * offset.x + target.heading.y * offset.y
    )
    const offsetTargetPos = Vec2.add(temp_v2, target.position, localOffset)
    const lookAheadTime: number = Vec2.distance(this._position, offsetTargetPos) / (this.maxSpeed + target.velocity.length())
    const predictedTarget: Vec2 =
        Vec2.add(
            temp2_v2,
            offsetTargetPos,
            Vec2.multiplyScalar(temp2_v2, target.velocity, lookAheadTime),
        );
    this.arrive(predictedTarget)
}

群落 flock

群落行为是由分离、凝聚和队列组成。

  • 分离(separation):每个角色都试着和相邻角色保持一定的距离。
  • 凝聚(cohesion):每个角色尽量不掉队,不落下太远。
  • 队列(alignment):每个角色尽可能与相邻角色行动于同一方向。

近了就离开,在一定范围内就靠近,速度取平均值。

flock(vehicles: Vehicle[]): void {
    let averageVelocity: Vec2 = temp3_v2.set(this.velocity)
    let averagePosition: Vec2 = temp4_v2.set(00)
    let inSightCount = 0;
    for (let i = 0; i < vehicles.length; i++) {
        let vehicle: Vehicle = vehicles[i] as Vehicle;
        if (vehicle != this && this.inSight(vehicle)) {
            averageVelocity = averageVelocity.add(vehicle.velocity);
            averagePosition = averagePosition.add(vehicle.position);
            if (Vec2.squaredDistance(this.position, vehicle.position) < this.tooCloseDist * this.tooCloseDist) this.flee(vehicle.position);
            inSightCount++;
        }
    }
    if (inSightCount > 0) {
        averageVelocity.multiplyScalar(1 / inSightCount);
        averagePosition.multiplyScalar(1 / inSightCount);
        this.seek(averagePosition);
        const force = averageVelocity;
        this._steeredForce.subtract(force);
    }
}

小结

转向行为的实现可分解为模拟各个行为的力,再计算合力,接着算加速度和速度,最后更新位置!

以上为白玉无冰使用 Cocos Creator 3.1.0 实现 "转向行为!" 的技术分享。

Cocos Store 链接:   http://store.cocos.com/app/detail/2893

参考资料

https://www.red3d.com/cwr/steer/
《Flash ActionScript 3.0 动画高级教程》

《游戏人工智能编程案例精粹》

keep hungry! keep foolish!

更多

折纸效果
竖直布局的文本
弹性跟随相机!
标志板!
2D 素材 3D 效果!
2020 原创精选!


更多精彩欢迎关注微信公众号



浏览 42
点赞
评论
收藏
分享

手机扫一扫分享

举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

举报