游戏开发常见算法

1.根据权重获取不同的值:
算法思想:
代码实现:
_proto.randWeightEnemy = function (enemyIdMap, enemyIds, targetWeight, weightArray, monsterNumLimit) {
   console.log("目标权重值"+targetWeight); //targetWeight的值为1700
   var rd = utils.intRange(1, targetWeight + 1);
   var index = 0;
   for (var j = 0; j < weightArray.length; j++) { //weightArray=[1000,1500,1700]
      if (rd < weightArray[j]) {
           index = j;
           break;
      }
    }
    enemyId = enemyIds[index]; //根据不同权重的下标一一对应不同id
    if (!enemyIdMap[enemyId]) {
       enemyIdMap[enemyId] = 0;
    }
    //当前怪物id数量+1存为临时数量
    var tmpCount = enemyIdMap[enemyId] + 1;
    if (tmpCount > monsterNumLimit[index]) {
    //超出限制
      enemyId = this.randWeightEnemy(enemyIdMap, enemyIds, targetWeight, weightArray, monsterNumLimit);  //嵌套执行
    } else {
      enemyIdMap[enemyId]++;
    }
      return enemyId; //不同id对应不同的权重值
};

2.自由落体:
(1)数据初始化定义:
//自由落体 0.5*a*t^2
//方向
//加速度 a
this.a = 1; //图片的像素移动 像素/ms
//初速度 v0 vt 当前速度 = v0 + at
this.v0 = 0;

(2)求自由落体 物体 y轴 坐标值的变化值:
/**
*求自由落体 Y轴 坐标值
* @param t0 帧时间差
*/
_proto.freeFall = function (fk, t1, t0) {    //fk  是做自由落体的物体,t1 是上一帧动画所在的时间,t0是帧时间差
    if(!fk)return;
    //Math.pow(2,3) 8 js封装的求平方数的方法
    fk.y += (this.v0 + this.a * t1 + 0.5 * this.a * t0) * t0;
}

(3)更新自由落体动画:
/**
* 更新下落动画
*/
_proto.updateFallAnim = function () {
    if(!this.isPlayingFallAnim)return;
    var fkIndex, row, vFk;
    console.log("下落清单");
    console.log(this.fall_list);
    var deltaTime = Laya.timer.delta;//帧时间差ms
    //单个方块移到目标点 完成单个动画播放 移除动画清单单个方块
    var l = this.fall_list.length;
    for (fkIndex = 0; fkIndex < l; fkIndex++) {
        vFk = this.fall_list[fkIndex];
        console.log("可以下落的方块");
        console.log(vFk.fallMaxY);
        row = this._getRow(vFk.fallMaxY);
        this.markRow(vFk, row); //只是做了数据的标记和存储,没有做表现
        this.freeFall(vFk, this.fallAnimTime, deltaTime);
        if(vFk.y >= vFk.fallMaxY){                 //  vFk.fallMaxY 是做自由落体物品的目标Y值
            this.fall_list.remove(vFk);            //单个下落结束后,就把结束下落的方块从清单中移除
            this.onSingleFallAimCompelete(vFk);
            fkIndex--;                             //index 依赖于 l l必须更严谨,因为它被index依赖,所以l-- 放在 index--后边
            l--;                                   //重要性高,放后边处理,覆盖前边
        }
    }
    this.fallAnimTime += deltaTime;//记录已播动画时长
}

3.角度和弧度的转换:
 
公式为:角度=180°×弧度÷π   弧度=角度×π÷180°

/** 弧度转角度换算单位 */
GameSetting$.ANGLE_1_RAD = 180 / Math.PI;
/** 角度转弧度换算单位 */
GameSetting$.RAD_1_ANGLE = Math.PI / 180;
4.两点之间距离的算法:
dx = Math.abs(p2.clientX - p1.clientX);
dy = Math.abs(p2.clientY - p1.clientY);
var dis = Math.sqrt(Math.pow(dx,2)+Math.pow(dy,2));

5.根据两点间坐标,计算方向向量,转化为角度:
        let dx = headPos.x - tailPos.x;
        let dy = headPos.y - tailPos.y;

        //2d方向向量
        let angle = Math.atan2(dy,dx);  //弧度
        this.targetAngle$ = angle * GameSetting$.ANGLE_1_RAD-90;

6.已知起点坐标,角度,长度,求终点坐标:
    /**
     * 已知起点,角度,长度,求终点坐标
     * @param {*} startPoint
     * @param {*} angle 角度
     * @param {*} distance
     * @returns
     */
    static calNewPointByAngle2$(startPoint, angle, distance) {
        var endPoint = {};
        // 角度转弧度
        var radian = (angle * Math.PI) / 180;
        // 计算新坐标(对于无限接近0的数字,此处没有优化)
        endPoint.x = startPoint.x + distance * Math.sin(radian);
        endPoint.y = startPoint.y + distance * Math.cos(radian);
        return endPoint;
    }

7.瞄准器脚本:鼠标移动画出2d的可控制角度的瞄准线
import GameSetting$ from "../../../Config/GameSetting";
import EventManager$ from "../../../Manager/EventManager";
import SSEVENT$ from "../../../Config/SSEVENT";
/**
 * 触摸区域画轨迹线2d
 * created by tcy 20210826
 */
export default class TouchDrawPoint2D$ extends Laya.Script {
    constructor() {
        super();
        this._initData$();
    }

    _initData$() {
        /**鼠标按下的点 */
        this._touchDownPos$ = null;
        /**鼠标区域box */
        this._box$ = null;
        /** 鼠标移动的方向 */
        this._direction$ = new Laya.Vector2(0, 0);
        /** 贴图路径 */
        this.textureUrl$ = GameSetting$.LINE_TEXTURE_URL$;
        /** 最大点数 */
        this.maxPointCnt$ = GameSetting$.MAX_LINE_POINT_CNT$;
        /**点的数量 */
        this.pointCnt$ = 0;
        /** 当前点的集合 */
        this._curPoint$ = [];
        /** Y轴方向点的间隔距离*/
        this._pointDisY$ = 50;
        /** 临时点变量 */
        this._tempV2$ = new Laya.Point();
        /** 临时点变量2 */
        this._tempV2_2$ = new Laya.Point();

    }

    onAwake() {
        this._box$ = this.owner;
    }

    onEnable() {
        this.owner.on(Laya.Event.MOUSE_DOWN, this, this._onTouchDown$);
        this.owner.on(Laya.Event.MOUSE_MOVE, this, this._onTouchMove$);
        this.owner.on(Laya.Event.MOUSE_UP, this, this._onTouchUp$);
        this.owner.on(Laya.Event.MOUSE_OUT, this, this._onTouchOut$);
    }

    onStart() {

        this.loadResArray$();
    }

    /**
    * 加载资源清单
    */
    loadResArray$() {
        Laya.loader.load(this.textureUrl$, Laya.Handler.create(this, this.onResLoaded$));
    }

    /**
     * 当资源加载完成
     */
    onResLoaded$() {
        /** 贴图 */
        this.texture$ = Laya.loader.getRes(this.textureUrl$);
    }

    _onTouchDown$(e) {

        let t = this._box$.getMousePoint();
        this._touchDownPos$ = new Laya.Point(t.x, t.y);
        this.graphicsSp$ = this.owner.addChild(new Laya.Sprite());
        this.graphicsSp$.x = this._touchDownPos$.x;
        this.graphicsSp$.y = this._touchDownPos$.y;
        this.drawSingleLine$(new Laya.Point(0, 0), 0);
    }

    _onTouchMove$(e) {
        if (!this.graphicsSp$ || this.graphicsSp$.destroyed) return;
        let t = this._box$.getMousePoint();
        if (!this._touchDownPos$ || this._touchDownPos$.x === t.x && this._touchDownPos$.y === t.y) return false;
        let i = new Laya.Point(t.x - this._touchDownPos$.x, t.y - this._touchDownPos$.y);
        this._tempV2$.copy(i);
        this._tempV2$.normalize();
        this._direction$.setValue(this._tempV2$.x, this._tempV2$.y);
        let dis = this._touchDownPos$.distance(t.x, t.y);
        this._initRotation$ = Math.atan2(this._direction$.y, this._direction$.x);
        //点数越多,点与点之间的间隔越大
        if (!this.isFulll$() && dis > 50) {
            let pointNum = Math.max(1, Math.floor(dis / 50));
            for (var g = 1; g <= pointNum; g++) {
                let point = new Laya.Point(0, this._pointDisY$ * g);
                if (this._curPoint$.indexOf(point.y) != -1) continue;
                this.drawSingleLine$(point, g);
            }

        }
        this.graphicsSp$ && (this.graphicsSp$.rotation = this._initRotation$ * GameSetting$.ANGLE_1_RAD$);
        this.resetPoint$(dis);
    }

    resetPoint$(dis) {
        if (!this.graphicsSp$ || this.graphicsSp$.destroyed) return;
        let pointNum = this.graphicsSp$._children.length;
        let pointNode;
        let index = Math.max(1, Math.floor(dis / 50));
        let disY = this.pointCnt$ * 3 || 3;
        for (var i = 0; i < pointNum; i++) {
            pointNode = this.graphicsSp$._children[i];
            let point = new Laya.Point(0, disY * i++);
            let pointNew = this.getBPoint$(point, -this._initRotation$);
            pointNode.x = pointNew.x;
            pointNode.y = pointNew.y;
            pointNode.index$ > index ? pointNode.visible = false : pointNode.visible = true;
        }
    }



    _onTouchUp$(e) {
        if (!this._touchDownPos$) return;
        EventManager$.getInstance$().dispatchEvent$(SSEVENT$.PLAYER_JUMP$, [this._direction$, this.pointCnt$])
        this.clearLines$();
    }

    _onTouchOut$() {
        this.clearLines$();
    }

    getDirection$() {
        return this._direction$;
    }

    /**
     * 清除所有的线
     */
    clearLines$() {
        this.clearLineDatas$();
        this.clearLinesGraphic$();
    }

    /**
    * 清除线数据
    */
    clearLineDatas$() {
        this.pointCnt$ = 0;
        this._curPoint$ = [];

    }

    /**
    * 点数是否上限
    */
    isFulll$() {
        return this.maxPointCnt$ <= this.pointCnt$;
    }

    /**
     * 
     * @param {*} source 
     * @param {*} angle Angle为正时逆时针转动, 单位为弧度
     * @returns 
     */
    getBPoint$(source, angle) {
        let R;
        let rotation = Math.atan2(this._direction$.y, this._direction$.x);
        rotation += angle//旋转
        R = Math.sqrt(source.x * source.x + source.y * source.y)//半径
        this._tempV2_2$.x = Math.cos(rotation) * R;
        this._tempV2_2$.y = Math.sin(rotation) * R;
        return this._tempV2_2$;
    }

    /**
    * 画单条线
    */
    drawSingleLine$(point, index) {
        this.pointCnt$++;
        this._curPoint$.push(point.y);
        // let pointNew = this.getBPoint$(point, -this._initRotation$);
        let newPoint = this.graphicsSp$.addChild(new Laya.Image(this.textureUrl$));
        newPoint.index$ = index;
        // newPoint.x = pointNew.x;
        // newPoint.y = pointNew.y;
    }

    /**
    * 清除线条图像
    */
    clearLinesGraphic$() {
        this.graphicsSp$ && this.graphicsSp$.destroy();
    }

    onDestroy() {
        Laya.timer.clearAll(this);
        this.owner.off(Laya.Event.MOUSE_DOWN, this, this._onTouchDown$);//按下
        this.owner.off(Laya.Event.MOUSE_UP, this, this._onTouchUp$);         //弹起
        this.owner.off(Laya.Event.MOUSE_MOVE, this, this._onTouchMove$);//移动鼠标
        this.owner.off(Laya.Event.MOUSE_OUT, this, this._onTouchOut$);
    }
}

  • 8
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
算法一:A*寻路初探 From GameDev.net 译者序:很久以前就知道了A*算法,但是从未认真读过相关的文章,也没有看过代码,只是脑子里有个模糊的概念。这次决定从头开始,研究一下这个被人推崇备至的简单方法,作为学习人工智能的开始。 这 篇文章非常知名,国内应该有不少人翻译过它,我没有查找,觉得翻译本身也是对自身英文水平的锻炼。经过努力,终于完成了文档,也明白的A*算法的原理。毫 无疑问,作者用形象的描述,简洁诙谐的语言由浅入深的讲述了这一神奇的算法,相信每个读过的人都会对此有所认识(如果没有,那就是偶的翻译太差了-- b)。 原文链接:http://www.gamedev.net/reference/articles/article2003.asp 以下是翻译的正文。(由于本人使用ultraedit编辑,所以没有对原文中的各种链接加以处理(除了图表),也是为了避免未经许可链接的嫌疑,有兴趣的读者可以参考原文。 会者不难,A*(念作A星)算法对初学者来说的确有些难度。 这篇文章并不试图对这个话题作权威的陈述。取而代之的是,它只是描述算法的原理,使你可以在进一步的阅读中理解其他相关的资料。 最后,这篇文章没有程序细节。你尽可以用任意的计算机程序语言实现它。如你所愿,我在文章的末尾包含了一个指向例子程序的链接。 压缩包包括C++和Blitz Basic两个语言的版本,如果你只是想看看它的运行效果,里面还包含了可执行文件。 我们正在提高自己。让我们从头开始。。。 序:搜索区域 假设有人想从A点移动到一墙之隔的B点,如下图,绿色的是起点A,红色是终点B,蓝色方块是中间的墙。 [图1] 你 首先注意到,搜索区域被我们划分成了方形网格。像这样,简化搜索区域,是寻路的第一步。这一方法把搜索区域简化成了一个二维数组。数组的每一个元素是网格 的一个方块,方块被标记为可通过的和不可通过的。路径被描述为从A到B我们经过的方块的集合。一旦路径被找到,我们的人就从一个方格的中心走向另一个,直 到到达目的地。 这些中点被称为“节点”。当你阅读其他的寻路资料时,你将经常会看到人们讨论节点。为什么不把他们描述为方格呢?因为有可 能你的路径被分割成其他不是方格的结构。他们完全可以是矩形,六角形,或者其他任意形状。节点能够被放置在形状的任意位置-可以在中心,或者沿着边界,或 其他什么地方。我们使用这种系统,无论如何,因为它是最简单的。 开始搜索 正如我们处理上图网格的方法,一旦搜索区域被转化为容易处理的节点,下一步就是去引导一次找到最短路径的搜索。在A*寻路算法中,我们通过从点A开始,检查相邻方格的方式,向外扩展直到找到目标。 我们做如下操作开始搜索: 1,从点A开始,并且把它作为待处理点存入一个“开启列表”。开启列表就像一张购物清单。尽管现在列表里只有一个元素,但以后就会多起来。你的路径可能会通过它包含的方格,也可能不会。基本上,这是一个待检查方格的列表。 2,寻找起点周围所有可到达或者可通过的方格,跳过有墙,水,或其他无法通过地形的方格。也把他们加入开启列表。为所有这些方格保存点A作为“父方格”。当我们想描述路径的时候,父方格的资料是十分重要的。后面会解释它的具体用途。 3,从开启列表中删除点A,把它加入到一个“关闭列表”,列表中保存所有不需要再次检查的方格。 在这一点,你应该形成如图的结构。在图中,暗绿色方格是你起始方格的中心。它被用浅蓝色描边,以表示它被加入到关闭列表中了。所有的相邻格现在都在开启列表中,它们被用浅绿色描边。每个方格都有一个灰色指针反指他们的父方格,也就是开始的方格。 [图2] 接着,我们选择开启列表中的临近方格,大致重复前面的过程,如下。但是,哪个方格是我们要选择的呢?是那个F值最低的。 路径评分 选择路径中经过哪个方格的关键是下面这个等式: F = G + H 这里: * G = 从起点A,沿着产生的路径,移动到网格上指定方格的移动耗费。 * H = 从网格上那个方格移动到终点B的预估移动耗费。这经常被称为启发式的,可能会让你有点迷惑。这样叫的原因是因为它只是个猜测。我们没办法事先知道路径的长 度,因为路上可能存在各种障碍(墙,水,等等)。虽然本文只提供了一种计算H的方法,但是你可以在网上找到很多其他的方法。 我们的路径是通过反复遍历开启列表并且选择具有最低F值的方格来生成的。文章将对这个过程做更详细的描述。首先,我们更深入的看看如何计算这个方程。 正 如上面所说,G表示沿路径从起点到当前点的移动耗费。在这个例子里,我们令水平或者垂直移动的耗费为10,对角线方向耗费为14。我们取这些值是因为沿对 角线的距离是沿水平或垂直移动耗费的的根号2(别怕),或者约1.414倍。为了简化,我们用10和14近似。比例基本正确,同时我们避免了求根运算和小 数。这不是只因为我们怕麻烦或者不喜欢数学。使用这样的整数对计算机来说也更快捷。你不就就会发现,如果你不使用这些简化方法,寻路会变得很慢。 既然我们在计算沿特定路径通往某个方格的G值,求值的方法就是取它父节点的G值,然后依照它相对父节点是对角线方向或者直角方向(非对角线),分别增加14和10。例子中这个方法的需求会变得更多,因为我们从起点方格以外获取了不止一个方格。 H 值可以用不同的方法估算。我们这里使用的方法被称为曼哈顿方法,它计算从当前格到目的格之间水平和垂直的方格的数量总和,忽略对角线方向。然后把结果乘以 10。这被成为曼哈顿方法是因为它看起来像计算城市中从一个地方到另外一个地方的街区数,在那里你不能沿对角线方向穿过街区。很重要的一点,我们忽略了一 切障碍物。这是对剩余距离的一个估算,而非实际值,这也是这一方法被称为启发式的原因。想知道更多?你可以在这里找到方程和额外的注解。 F的值是G和H的和。第一步搜索的结果可以在下面的图表中看到。F,G和H的评分被写在每个方格里。正如在紧挨起始格右侧的方格所表示的,F被打印在左上角,G在左下角,H则在右下角。 [图3] 现在我们来看看这些方格。写字母的方格里,G = 10。这是因为它只在水平方向偏离起始格一个格距。紧邻起始格的上方,下方和左边的方格的G值都等于10。对角线方向的G值是14。 H 值通过求解到红色目标格的曼哈顿距离得到,其中只在水平和垂直方向移动,并且忽略中间的墙。用这种方法,起点右侧紧邻的方格离红色方格有3格距离,H值就 是30。这块方格上方的方格有4格距离(记住,只能在水平和垂直方向移动),H值是40。你大致应该知道如何计算其他方格的H值了~。 每个格子的F值,还是简单的由G和H相加得到 继续搜索 为了继续搜索,我们简单的从开启列表中选择F值最低的方格。然后,对选中的方格做如下处理: 4,把它从开启列表中删除,然后添加到关闭列表中。 5,检查所有相邻格子。跳过那些已经在关闭列表中的或者不可通过的(有墙,水的地形,或者其他无法通过的地形),把他们添加进开启列表,如果他们还不在里面的话。把选中的方格作为新的方格的父节点。 6,如果某个相邻格已经在开启列表里了,检查现在的这条路径是否更好。换句话说,检查如果我们用新的路径到达它的话,G值是否会更低一些。如果不是,那就什么都不做。 另一方面,如果新的G值更低,那就把相邻方格的父节点改为目前选中的方格(在上面的图表中,把箭头的方向改为指向这个方格)。最后,重新计算F和G的值。如果这看起来不够清晰,你可以看下面的图示。 好了,让我们看看它是怎么运作的。我们最初的9格方格中,在起点被切换到关闭列表中后,还剩8格留在开启列表中。这里面,F值最低的那个是起始格右侧紧邻的格子,它的F值是40。因此我们选择这一格作为下一个要处理的方格。在紧随的图中,它被用蓝色突出显示。 [图4] 首先,我们把它从开启列表中取出,放入关闭列表(这就是他被蓝色突出显示的原因)。然后我们检查相邻的格子。哦,右侧的格子是墙,所以我们略过。左侧的格子是起始格。它在关闭列表里,所以我们也跳过它。 其 他4格已经在开启列表里了,于是我们检查G值来判定,如果通过这一格到达那里,路径是否更好。我们来看选中格子下面的方格。它的G值是14。如果我们从当 前格移动到那里,G值就会等于20(到达当前格的G值是10,移动到上面的格子将使得G值增加10)。因为G值20大于14,所以这不是更好的路径。如果 你看图,就能理解。与其通过先水平移动一格,再垂直移动一格,还不如直接沿对角线方向移动一格来得简单。 当我们对已经存在于开启列表中的4个临近格重复这一过程的时候,我们发现没有一条路径可以通过使用当前格子得到改善,所以我们不做任何改变。既然我们已经检查过了所有邻近格,那么就可以移动到下一格了。 于 是我们检索开启列表,现在里面只有7格了,我们仍然选择其中F值最低的。有趣的是,这次,有两个格子的数值都是54。我们如何选择?这并不麻烦。从速度上 考虑,选择最后添加进列表的格子会更快捷。这种导致了寻路过程中,在靠近目标的时候,优先使用新找到的格子的偏好。但这无关紧要。(对相同数值的不同对 待,导致不同版本的A*算法找到等长的不同路径。) 那我们就选择起始格右下方的格子,如图。 [图5]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小春熙子

你一毛我一毛,先富带后富

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值