php实现贪食蛇游戏,使用js实现贪吃蛇游戏的代码

这篇文章主要介绍了关于使用js实现贪吃蛇游戏的代码,有着一定的参考价值,现在分享给大家,有需要的朋友可以参考一下13行常规(700bytes)shortest snake game.html

压缩后的500bytes(当然两处document还是可以用eval压缩的)index.500bytes.html

之前很火的20行代码地址(有BUG)(900bytes)hj7jay/article/details/51011269

一维数组700char

(0,0)位置的蛇身用0表示,(0,1)用1,(1,0)用10表示,以此类推

因为就13行js,

第4行 是声明

第5行 比较难理解,可以把?:运算符,拆分为4行if语句;可以参考下面的 二维数组 的版本

第9行0|x 和 ~~x 和 x>>0 都能去除x(number)的尾数

应该算易读了

let dir=1,food=3,snk=[1,0],ctx=document.getElementById("1").getContext("2d")

document.οnkeydοwn=e=>{dir = snk[0]-snk[1]==-(tmp = [-1,-10,1,10][e.keyCode-37]||dir)?dir:tmp }

setInterval(()=>{

snk.unshift(Head = snk[0] + dir)

if(Head!=food) snk.pop()

else while(snk.includes(food=0|Math.random()*10*10)) ;

if(snk.indexOf(Head,1)!=-1||(dir==1&&Head%10==0)||(dir==-1&&Head%10==9)||Head<0||Head>=100)

return document.write(0&snk.shift()) //死亡记录蛇长

for(let i=0; i<100; i++){

ctx.fillStyle = '#0'+(food==i)*9910+snk.includes(i)*1990

ctx.fillRect(i%10*40,(i-i%10)*4, 40,40)

}

},100)

颜色效果

let dir=1,food=3,snk=[1,0],n_=0, ctx=document.getElementById("1").getContext("2d")

document.οnkeydοwn=e=>{ dir =snk[0]-snk[1]==-(tmp = [-1,-10,1,10][e.keyCode-37]||dir)?dir:tmp }

setInterval(()=>{

snk.unshift(Head = snk[0] + dir)

if(Head!=food) snk.pop()

else while(snk.includes(food=0|Math.random()*10*10)) ;

if(snk.indexOf(Head,1)!=-1||(dir==1&&Head%10==0)||(dir==-1&&Head%10==9)||Head<0||Head>=100)

return alert("died"+ ++n_+"times") //死亡记录死亡次数

for(let i=0 ; i<100; i++){

ctx.fillStyle = '#0'+~~((food===i)*13000*Math.random())+~~(snk.includes(i)*3000*Math.random())

ctx.fillRect(i%10*40,(i-i%10)*4, 40,40)

}

},120)

说明

如果自己要写的话:要注意两点蛇尾应该比蛇头先消失,蛇头应该比食物先生成,

蛇不能走当前相反的方向

可以用长度为4的蛇进行测试

代码风格省去没必要的标签

https://google.github.io/styleguide/htmlcssguide.html#Optional_Tags

if() return只用一行

https://google.github.io/styleguide/cppguide.html

除了键盘响应那里用3目运算符能省3行之外,其他地方都没必要用

二维数组

以上的都是用一维数组实现的,下面的用二维数组写;要简化也能简化到17行以内900char以内(比20行的那个短就是了),不过没有必要

带注释的1100char

ctx = document.getElementById("1").getContext("2d") //CanvasRenderingContext2D inferface

let Len = 10, dir = 2, dirNow ; //dirNow 后面解释

food = [3, 0]; Snake = [[0, 0], [1, 0]] //食物的坐标,蛇身的坐标用Snake数组记录

Map = {'0,0':'#52a', '1,0':'#52a'} //用来记录绘图颜色的 地图

dirMat = [[-1, 0], [0, -1], [1, 0], [0, 1]] //方向矩阵

pairEq = ((p1, p2) => p1[0] == p2[0] && p1[1] == p2[1]) //检测 两数对 是否相等的函数

document.onkeydown = e =>{

if (37 <= e.keyCode == e.keyCode < 41 && dirNow != ( (e.keyCode - 35) % 4) ) //确定是 方向键 并且 保证方向与当前运动方向相反

dir = e.keyCode -37

}

!function () {

Head = Snake[Snake.length-1].map((x, i) => x + dirMat[dirNow=dir][i]); //得到头部接下来的移动位置

if (!pairEq(Head, food))

Map[Snake.shift()]='#fff' //必须先删尾巴,才能加入头部,吃没吃到食物是唯一判断标准

if (Snake.some(x=>pairEq(x,Head)) || !Head.every(x => 0<=x == x < Len)) //判断蛇头是否撞到蛇身或墙壁

return document.write("Game Over") //这样调用document.write会把页面全部清空

Snake.push(Head); //可以加入头部

while (Snake.some(x => pairEq(x, food))) //加入新头后, 生成食物更方便

food = [~~(Math.random() * Len), ~~(Math.random() * Len)]; //因为js没有整形的概念, ~~ 现在相当于向原点舍去

Map[Head] = '#52a' ; Map[food] = '#ad5'

for( k in Map){

ctx.fillStyle= Map[k]

ctx.fillRect(parseInt(k[0])*40,parseInt(k[2])*40,40,40) //e.g. k="1,3",也因此地图大小限制为10

}

setTimeout(arguments.callee, 100); //100ms后调用此函数一次

}()

七彩的(主要是比较好看)1100char

ctx = document.getElementById("1").getContext("2d")

let Len = 10, dir = 2, dirNow ;

food = [3, 0]; Snake = [[0, 0], [1, 0]]

Map = {'0,0':'fff', '1,0':'fff'}

dirMat = [[-1, 0], [0, -1], [1, 0], [0, 1]]

pairEq = ((p1, p2) => p1[0] == p2[0] && p1[1] == p2[1])

document.onkeydown = e =>{

if (37 <= e.keyCode == e.keyCode < 41 && dirNow != ( (e.keyCode - 35) % 4) )

dir = e.keyCode -37

}

!function () {

Head = Snake[Snake.length-1].map((x, i) => x + dirMat[dirNow=dir][i]);

if (!pairEq(Head, food)) Map[Snake.shift()]='0'

if (Snake.some(x=>pairEq(x,Head)) || !Head.every(x => 0<=x == x < Len))

return document.write("Game Over")

Snake.push(Head);

while (Snake.some(x => pairEq(x, food)))

food = [~~(Math.random() * Len), ~~(Math.random() * Len)];

Map[Head] = Map[food] = 'fff'

for( k in Map){

ctx.fillStyle='#'+(0xfff-~~(parseInt(Map[k],16)*Math.random())).toString(16)

ctx.fillRect(parseInt(k[0])*40,parseInt(k[2])*40,46,43)

}

setTimeout(arguments.callee, 100);

}()

以上就是本文的全部内容,希望对大家的学习有所帮助,更多相关内容请关注PHP中文网!

相关推荐:

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值