0%

AStar+BST+canvas实现JS版自动寻路贪吃蛇

前言(2020-05-06更新)

线上预览地址 贪吃蛇自动寻路

游戏领域,寻路算法是一个老生常谈的问题,常见的DFS、BFS、A*、Dijkstra等等还有其它博主未知的算法,而贪吃蛇是一个很好的学习寻路算法的项目。

本文基于A*+BST+canvas的自动寻路贪吃蛇—–JS版,目前还没实现吃满算法。

很久之前就用JS写过A*自动寻路的贪吃蛇,不过就算直到目前为止还是没有写出吃满算法, 最近在补数据结构算法的知识,现学现用,算是重构之前的实现。

此BST非平衡BST。

本文代码传送门:A*+BST+canvas实现JS版贪吃蛇

2020-05-06更新

目前基本实现吃满,但是很费时间,走S形估计可以很快,不过下图算法又不像完全走S, 总之达不到下图俄罗斯大佬的优秀。暂时不折腾了,改日再会。

Yk5R0S.gif

效果图-最初版本

Y9AldA.gif

效果图-吃满版本

Yk5awD.gif

JS实现一个二叉搜索树

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
let Util  = {
debug:false,
log(msg){
if(Util.debug){
console.log(msg);
}
},
//注意没有实现深度拷贝
extend(){
var i =1, target = arguments[0],length = arguments.length,options,src,copy,name;
if(typeof target!="object" &&typeof target!="function"){
target = {};
}
for (;i<length;i++){
if((options=arguments[i])!=null){
for (name in options) {
src = target[ name ];
copy = options[ name ];
if ( target === copy ) {
continue;
}
target[ name ] = copy;
}
}
}
return target;
},
//JS版二叉搜索树
BST: function (compareFunc) {
var _this = this;
this.root = null;
this.compareFunc = compareFunc;
this.size = 0;
this.insert = function (data) {
if (isNaN(data) && !this.compareFunc) {
return;
}
this.root = insertEle(this.root, data);
this.size = this.size+1;
}
//局部函数也就相当于私有函数
var insertEle = function (node, data) {
if (node == null) {
return new Node(data);
}
if (!_this.compareFunc) {
if (data > node.data) {
node.right = insertEle(node.right, data);
} else if (data < node.data) {
node.left = insertEle(node.left, data);
} else {
node.num.push(data);
}
} else {
var comRes = _this.compareFunc(data, node.data);
if (comRes > 0) {
node.right = insertEle(node.right, data);
} else if (comRes < 0) {
node.left = insertEle(node.left, data);
} else {
node.num.push(data);
}
}
return node;
}

this.removeMin = function () {
if (this.root == null) return;
var min = this.findMin();
this.root = removeMinEle(this.root);
return min;
}

this.findMin = function () {
if(this.root==null) return null;
let min = findMinEle(this.root);
return min;
}



var findMinEle = function (node) {
if (node.left == null) {//说明此节点已经是最小的了
return node;
}
return findMinEle(node.left);
}


this.findMax = function () {
if(this.root==null) return null;
let max = findMaxEle(this.root);
return max;
}

var findMaxEle = function (node) {
if (node.right == null) {//说明此节点已经是最大的了
return node;
}
return findMaxEle(node.right);
}

var removeMinEle = function (node) {
if (node.left == null) {//说明此节点已经是最小的了
node.minData = node.num.pop();
_this.size = _this.size-1;
if (node.num.length > 0) {
return node;
}
var right = node.right;
node = node.right = null;
return right;
}
node.left = removeMinEle(node.left);
return node;
}

//中序遍历
this.midOrder = function () {
midOrderEle(this.root);
}

this.isEmpty = function(){
return this.size==0;
}

var midOrderEle = function (node) {
if (node == null) {
return;
}
midOrderEle(node.left);
console.log(node.data);
midOrderEle(node.right);
}

var Node = function (data) {
this.left = null;
this.right = null;
this.data = data;
this.num = [];
this.num.push(data);
}
}
}

export default Util

自动寻路逻辑

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
import Util from './util'

/***
* auth by yang
* email 1249492252@qq.com
* http://blog.yxyvpn.top/
* @param options
*/

function snake(options){
this.options =Util.extend(defaultOps,options) ;
this.opt = this.options;
this.data = [];
this.canvasDom = document.createElement("canvas");
this.closeMap = {};
//维护一个空白映射对象 表明这些位置是空白的 用来随机生成食物
this.open = {};
this.count = 0;
this.maxcount = 0;
this.followLast= false;
Util.debug = !!this.opt.debug;
this.init();
}


let snakeFunc = {
init: function () {
//初始化画板
var opt = this.opt, canvasDom = this.canvasDom;
canvasDom.style = opt.style;
var gameWidth = this.gameWidth = opt.xSize * opt.unitSize;
var gameHeight = this.gameHeight = opt.ySize * opt.unitSize ;
canvasDom.style.width = gameWidth+"px";
canvasDom.style.height = gameHeight+"px";
canvasDom.style.border = opt.border;
document.querySelector("body").append(canvasDom);
var canvas=this.canvas = canvasDom.getContext("2d");
//绘制不清晰 百度说要加下面这行
let pixelRatio = this.getPixelRatio(canvas);
canvas.width = gameWidth;
canvas.height = gameHeight;
canvas.canvas.width = gameWidth*pixelRatio;
canvas.canvas.height = gameHeight*pixelRatio;
//绘制不清晰 百度说要加下面这行
canvas.scale(pixelRatio, pixelRatio);
//随机生成头的位置
let head =this.head = this.randomPos();
this.data.push(head);


//根据头的位置随机生成方向 如果x==0 则方向不能往右,如果x==xSize-1 则不能往左, 其它方向同理
//randomOrientation已经做了方向的处理
let randomOrientation = this.randomOrientation();
var body;
switch (randomOrientation) {
case 0:
body = [head[0],head[1]+1];
break;
case 1:
body = [head[0]-1,head[1]];
break;
case 2:
body = [head[0],head[1]-1];
break;
case 3:
body = [head[0]+1,head[1]];
break;
}
this.data.push(body);

for (let i = 0; i < opt.ySize; i++) {
for (let j = 0; j < opt.xSize; j++) {
this.open[this.getKey([i,j])] = [i,j];
}
}

//维护关闭列表
this.addCloseMap(head);
this.addCloseMap(body);

this.food = this.createFood();
this.paint();
var _this = this;
setTimeout(function () {
_this.render(_this);
},33);
},
addCloseMap(pos){
this.closeMap[this.getKey(pos)] = true;
delete this.open[this.getKey(pos)];
},
removeCloseMap(pos){
delete this.closeMap[this.getKey(pos)];
this.open[this.getKey(pos)] = pos;
},
render: function (_this) {
let pos = null, maxcount = _this.maxcount;
let size = _this.data.length;
var last = _this.data[size - 1],head = _this.head,food = _this.food;
_this.removeCloseMap(last);
if(_this.followLast){
_this.count++;
pos = _this.findPerfectNode(head, last, true);
if(_this.count>=_this.maxcount){
_this.count=0;
_this.followLast = false;
Util.log("================count = "+_this.count)
}
}else{
var startTIme = Date.now();
pos = _this.findPerfectNode(head, food);
var endTime = Date.now();
Util.log("查找耗时:"+(endTime-startTIme)+"毫秒,成功="+(pos!=null));
if (pos != null) {
//使用这个位置看是否能找到自己的尾巴,如果不能则随机走一步
var lastFind = _this.findPerfectNode(pos, last);
_this.removeCloseMap(pos);
if (lastFind == null) {
pos = null;
Util.log("不能找到尾巴!");
}else{
Util.log("能找到尾巴!");
}
}
}
if(pos==null){
pos = _this.findPerfectNode(head, last, true);
Util.log("跟着尾巴走");
_this.followLast= true;
let num = _this.opt.xSize*_this.opt.ySize;
_this.maxcount = Math.floor(Math.random()*(num*0.25))+10;
}
_this.addCloseMap(last);
Util.log("closeMap=" + Object.keys(_this.closeMap).length + "" +
";size=" + _this.data.length + ";blank=" + Object.keys(_this.open).length);
if (pos == null) {
alert("游戏结束了!")
return;
}
_this.head = pos;
//删除最后一个元素 并且在前面添加一个元素
_this.data.unshift(pos);
_this.addCloseMap(pos);
if (_this.posEquals(food, pos)) {
_this.food = _this.createFood();
if (_this.food == null) {
alert("恭喜,吃满了!");
return;
}
} else {
_this.last = _this.data.pop();
_this.removeCloseMap(_this.last);

}
_this.paint();
if (window.requestAnimationFrame) {
window.requestAnimationFrame(function () {
_this.render(_this)
});
} else if (window.webkitRequestAnimationFrame) {
window.webkitRequestAnimationFrame(function () {
_this.render(_this)
});
} else {
setTimeout(function () {
_this.render(_this);
}, 33);
}
},
switchNext: function (type,findNode) {//废弃不用
//根据当前的位置选择一个离食物或者其它目标最远的位置走
var distantBst = new Util.BST(function (a, b) {
return a.value - b.value;
});
var head = this.head, resGH,closeMap = this.closeMap,food = this.food;
if(!findNode){
findNode = food;
}
var neighbors = [
[head[0], head[1] - 1],
[head[0], head[1] + 1],
[head[0] - 1, head[1]],
[head[0] + 1, head[1]]
];
var validPoss = [];
for (let i = 0; i < neighbors.length; i++) {
const neighbor = neighbors[i];
if(closeMap[this.getKey(neighbor)]) continue;
resGH = this.calcGH(neighbor, findNode);
if(resGH>=0){
distantBst.insert({
pos:neighbor,
value: resGH
});
validPoss.push(neighbor);
}
}
switch (type) {
case 1:
if(!distantBst.isEmpty()){
return distantBst.findMin().data.pos;
}
break;
case 3:
if(!distantBst.isEmpty()){
return distantBst.findMax().data.pos;
}
break;
case 2:
//随机位置
return validPoss[Math.floor(Math.random()*validPoss.length)];
break;
}
return null;
},
calcGH(src,des){//A*计算GH值
var x = src[0],y=src[1];
if(x<0||x>=this.opt.xSize||y<0||y>=this.opt.ySize) return;
x = des[0],y=des[1];
if(x<0||x>=this.opt.xSize||y<0||y>=this.opt.ySize) return;
return Math.abs(src[0]-des[0])+Math.abs(src[1]-des[1]);
},
createBST(flag){
//二叉搜索树比较函数 {value:A*算法G+H,pos:坐标}
if(flag){//距离最远
return new Util.BST(function (a,b) {
return b.value-a.value;
});
}
return new Util.BST(function (a,b) {//距离最近
return a.value-b.value;
});
},
paint(){//绘制函数
var opt = this.opt;
var canvas=this.canvas,xSize =opt.xSize,ySzie = opt.ySize,
unitSize = opt.unitSize,
data = this.data;
//清空画布
canvas.clearRect(0,0,xSize*unitSize,ySzie*unitSize);
//绘制线条
/*canvas.strokeStyle = "#ccc";
canvas.lineWidth = 1;
for (let i = 1; i < ySzie; i++) {
canvas.beginPath();
canvas.moveTo(i*unitSize,0);
canvas.lineTo(i*unitSize,ySzie*unitSize);
canvas.stroke();
canvas.closePath();
}
for (let j = 0; j < xSize; j++) {
canvas.beginPath();
canvas.moveTo(0,j*unitSize);
canvas.lineTo(xSize*unitSize,j*unitSize);
canvas.stroke();
canvas.closePath();
}*/
//绘制蛇
canvas.fillStyle=opt.headColor;
this.fillUnit(this.head);//头
canvas.fillStyle=opt.bodyColor;
for (let i = 1; i < data.length-1; i++) {
this.fillUnit(data[i]);
}
//绘制尾巴
canvas.fillStyle=opt.tailColor;
this.fillUnit(this.data[this.data.length-1]);
//绘制食物
canvas.fillStyle=opt.foodColor;
this.fillUnit(this.food);
},
fillUnit:function(pos){
var opt = this.opt;
var canvas=this.canvas,xSize =opt.xSize,ySzie = opt.ySize,
unitSize = opt.unitSize,
data = this.data;
let size = unitSize;
canvas.fillRect(pos[0]*unitSize,pos[1]*unitSize,size+1,size+1);
},
getPixelRatio(context) {
var backingStore = context.backingStorePixelRatio ||
context.webkitBackingStorePixelRatio ||
context.mozBackingStorePixelRatio ||
context.msBackingStorePixelRatio ||
context.oBackingStorePixelRatio ||
context.backingStorePixelRatio || 1;
return (window.devicePixelRatio || 1) / backingStore;
},
createFood:function(){
var blankKeys = Object.keys(this.open);
if(blankKeys.length==0){
return null;
}
let index = Math.floor(Math.random()*blankKeys.length);
let food = this.open[blankKeys[index]];
delete this.open[this.getKey(food)];
return food;
},
randomPos(){
var _this = this;
return [Math.floor(Math.random()*_this.opt.xSize),Math.floor(Math.random()*_this.opt.ySize)];
},
randomOrientation() {
// 1 上 2右 3 下 4 左
let head = this.data[0];
let x = head[0], y = head[1];
let wz = [1,1,1,1];
if(x==0) wz[1] = 0;
else if(x==this.opt.xSize-1) wz[3] = 0;

if(y==0) wz[2] = 0;
else if(y==this.opt.ySize-1) wz[0] = 0;

let wz2 = [];
for (let i = 0; i < wz.length; i++) {
let w = wz[i];
if(w==1){
wz2.push(i);
}
}
return wz2[Math.floor(Math.random()*wz2.length)];
},
findPerfectNode:function(startNode,findNode,flag){//根据A*贪心算法试图寻找局部最优以至全局最优的路径
this.bst = this.createBST(flag);
var perfectNode = null,closeMap = {},pos,key,tcloseMap = this.closeMap;
this.removeCloseMap(startNode);
this.putBST(startNode,findNode);
while(!this.bst.isEmpty()){
let node = this.bst.removeMin();
perfectNode = node.minData;
pos = perfectNode.pos;
key = this.getKey(pos);
if(closeMap[key]||tcloseMap[key]) continue;
//将该节点周围四个节点加入bst
this.putBST([pos[0],pos[1]-1],findNode,perfectNode);//上
this.putBST([pos[0],pos[1]+1],findNode,perfectNode);//下
this.putBST([pos[0]-1,pos[1]],findNode,perfectNode);//左
this.putBST([pos[0]+1,pos[1]],findNode,perfectNode);//右
if(this.posEquals(pos,findNode)){
//已经找到
break;
}
//加入关闭列表
closeMap[key] = true;
}
this.addCloseMap(startNode);
if(perfectNode==null||!this.posEquals(perfectNode.pos,findNode)){
Util.log("查找失败!");
return;
}
//回溯
while(perfectNode.parent){
var parent = perfectNode.parent;
if(parent.parent==null){
break;
}
perfectNode = parent;
}
return perfectNode.pos;
},
getKey(pos){
return "x="+pos[0]+":y="+pos[1];
},
posEquals(pos1,pos2){
if(pos1[0]==pos2[0]&&pos1[1]==pos2[1]) return true;
return false;
},
putBST(startNode,findNode,parent) {
var x = startNode[0],y = startNode[1];
if(x<0||x>=this.opt.xSize||y<0||y>=this.opt.ySize) return;
var d = {
parent:parent,
pos:startNode,
value:this.calcGH(startNode,findNode)
}
this.bst.insert(d);
}

};

snake.prototype = snakeFunc;

let defaultOps = {
debug:true,//是否开启日志,控制台打印运行状态
style:"margin:0 auto;background:#fff;display: block;",//画布样式
headColor:"pink",//头颜色
bodyColor:"#ccc",//身体颜色
foodColor:"red",//食物颜色
tailColor:"blue",//尾巴颜色
xSize:20,//矩阵长
ySize:20,//矩阵宽
unitSize:10,//单元格大小
border:"1px solid #ccc"//画布边框
}

export default snake
坚持原创技术分享,您的支持将鼓励我继续创作!
YANG 微信支付

微信支付

YANG 支付宝

支付宝