HTML5 游戏开发实战 | 贪吃蛇

2023-06-29 15:11:52

在该游戏中,玩家操纵一条贪吃的蛇在长方形场地里行走,贪吃蛇按玩家所按的方向键折行,蛇头吃到食物(豆)后,分数加10分,蛇身会变长,如果贪吃蛇碰上墙壁或者自身的话,游戏就结束了(当然也可能是减去一条生命)。

贪吃蛇游戏的运行界面如上图所示。

01、贪吃蛇游戏设计的思路

把游戏画面看成40×30的方格。食物(豆)和组成蛇的块均在屏幕上占据一个方格。游戏设计中主要用到的4个类如下。

Farm类:主要用来显示场地,随机生成食物,初始化一条蛇。

Food类:抽象了食物(豆)的属性和动作。

Snake类:抽象了贪吃蛇的属性和动作,调用Block类来组成蛇,并处理键盘输入事件和蛇的移动。

Block类:表示组成蛇的块(实心圆)。一条蛇可以看成由许多“块”(或称节)拼凑而成,块是蛇身上最小的单位。

02、贪吃蛇游戏设计的步骤

游戏页面index.html

<!DOCTYPE html>
<html lang = "en">
< head >
<meta charset ="UTF - 8"
< title>小游戏之贪吃蛇</title>
< style>
# canvas{border: 3px solid red;</style>
</head>
< body>
<canvas id='canvas'width=800'height =00'></canvas><div id="textmsg">分数</div>
</body 

设计脚本

1. 食物(豆)类(Food)设计

在此游戏中,首先会在场地的特定位置出现一个豆,豆要不断被蛇吃掉,当豆被吃掉后,原豆消失,又在新的位置出现新的豆。这些豆都是由豆(Food)类创建的对象。

foodInit()函数用于在屏幕上显示一个豆(实心圆),设计方法是直接在场地(canvas)上画一个实心圆。

equal()函数用于判断是否与蛇身“块”node重合,也就是蛇吃到食物。

//食物类
function Food(x,y,w) {
var t= this;
t.x=x;
t.y=y;
t.w=w;//食物
//X坐标
//Y坐标
//大小
t.foodInit = function() [
//画一个实心圆
ctx.beginPath();
ctx.arc(x+w/2,y+w/2,w/2,0,360,false);ctx.fillStyle="red";
//填充颜色,默认是黑色
//画实心圆
ctx.fill();
ctx.closePath();
//判断是否重合
t.equal = function(node) 
if(this.x == node.x && this.y== node.y) return true;else(
return false;
}
}
}

 

2. 块类(Block)

在贪吃蛇游戏中,块用来构成蛇,在蛇出现时,要把构成蛇的块一个个地输出(显示),在蛇消失时,要把块消除掉,显示和消除哪一个块都要由位置决定,并且由于蛇是由多个块构成的,每个块要填到snakes数组中。

//蛇块类
function Block(x,Y,w)(
var t = this;
t.x= x;
t.y=y;
t.w= w;
//画一个蛇块
t.drawBlock = function()
ctx.beginPath();
ctx.arc(x+w/2,y+w/2,w/2,0,360,false);
//填充颜色,默认是黑色
ctx.fillStyle="blue";
ctx.fill();
//画实心圆
//清除蛇块
t.clear = function()!
ctx.fillStvle=white';
ctx.strokeStyle = white';
ctx.fillRect(x,Y,w,w);
ctx.strokeRect(x,Y, w,w);
//判断是否重合
t.equal = function(node)
if(this.x== node.x && this.y== node.y){return true;
else
return false;
}
}

 

3. 蛇类(Snake)设计

现在到了最难的步骤,就是处理蛇,一条完整的贪吃蛇是由一块一块组成的。snakes数组用于存放组成蛇的所有块;其中保存的第一个元素是蛇的头部,最后一个元素是蛇的尾巴。当蛇运动的时候,它头部增加一块而尾部减少一块。如果它吃到了豆,头部增加一块而尾部不减少。也就是说,蛇是从头部开始长的。蛇运行过程中要不断地改变方向;如果蛇头碰到了它自身,蛇就要死亡,即程序结束。

首先,画一条蛇并移动它。

//蛇类
function Snake(x,y  len,speed) {
var t = this;
t.x=x;
t.y=y;
t.dir='R';
t.len = len;
//dir 方向,R'向右
var nx = x;ny =y;
//初始蛇最初 len(5)块,并启动定时
t.init = function()
for(var i=0;i< len; i++){
var tempBlock = new Block(nx,ny,gridWidth);tempBlock.drawBlock();nx-= gridWidth;
snakes.push(tempBlock);
snake interval = setInterval(t.move,speed);//定时移动蛇
}

 然后,识别键盘事件,修改移动方向dir,初始移动dir方向为'R'(向右)。

//取得键盘方向
document.onkevdown = function(e)
var code = e.keyCode;
t.odir = t.dir;
switch(code)
case 37:
t.dir='L';
break;case 38:
t.dir='u';break;
case 39:
t.dir='R';break;case 40 :
t.dir='D';
//向左键
//向上键
//向右键
//向下键
break;
}
}

 

以下主要是让蛇动的move()函数。主要是根据原来蛇头snakes[0]的位置和移动方向确定新的蛇头位置,绘制新的蛇头,并清除原来的蛇尾即达到移动效果。

在蛇移动时,判断蛇头是否和食物相撞,是否碰撞到了场地的壁以及是否与自己相撞。

//移动蛇
t.move = function()
var newHead;
//是否碰撞到了场地的壁
if(snakes0].x+ snakes0].w >= canvas.width snakes0].x- snakes0].w<0snakes[0].y- snakes[0].w < 0 snakes[0].y + snakes[0].w > canvas.height)(gameover();
else{//根据原来蛇头 snakes[0]的位置和移动方向确定新的蛇头位置
if(t.dir=='R'){newHead= new Block(snakes[0].x + gridWidth,snakes[0].y,gridWidth);) else if (t.dir =='L')(
newHead = new Block(snakes[0].x- gridWidth,snakes[0].y,gridWidth);else if (t.dir==D') !
newHead = new Block(snakes[0].x,snakes[0].y + gridWidth,gridWidth);else if (t.dir==u') !
newHead= new Block(snakes[0].x,snakes[0].y- gridwidth,gridWidth)
//禁止反向跑
if(newHead.x == snakes[1].x && newHead.y == snakes[1].y)t.dir = t.odir;return;
//画新的蛇头newHead.drawBlock();//追加到数组中(长度会自动加)snakes.unshift(newHead);//清除原来尾部snakes[snakes.length - 17.clear();//并从数组中移除(长度会自动减)snakes.pop()
//清除(蛇尾)块
/判断食物是否和蛇头相撞
for(var i=0;i< foods.length; i++)if(foods[ i].equal( snakes[0]))
//给蛇增加长度
t.growth();
score= score +10:
textmsq.innerHTML = score +“分”
t.len = t.len + 1;
//蛇生长方法
//增加 10 分
//显示分数
clearInterval(snake interval);
//速度加快snake interval = setInterval(t.move,speed);
speed = speed < 20 ? speed : speed -10;
//判断是否与自己相撞
for(var i=1;i< snakes.length; i++)if(snakesil.equal(snakes[0T)) gameover();
)//move()函数结束

用于实现蛇生长growth()函数的具体功能是当蛇吃到一个豆后,蛇就要在它的尾巴上增加一块即蛇增长。设计思路是找到蛇尾snakes[snakes.length-1],根据蛇尾与蛇的倒数第2块snakes[snakes.length-2]的位置关系,计算出蛇尾新增一块的位置。

 

//给蛇增加长度(在尾巴加)
t.growth = function()
 var tail1 = snakes[ snakes.length - 1];
var tai12 = snakes[snakes.ength - 2];
var addBlock;
if(tai11.x== tai12.x) 
{
if(tail1.y>= tail2.y)
}
addBlock =new Block(tail1.x,tail1.y + gridWidth,gridWidth);
else
addBlock =new Block(tail1.x,tail1.y-gridWidth,gridWidth);
else
if(tai11.x>= tail2.x)
addBlock =new Block(tail1.x + gridWidth,tail1.y,gridWidth);
else
addBlock =new Block(tail1.x-gridWidth,tail1.y,gridWidth);
//数组加入尾部
snakes.push(addBlock);addBlock.drawBlock();
console.log(snakes.length);
)//growth()函数
* snake 类结束 */

 

4. 场地类(Farm)设计

为游戏的主场地,豆要在此范围内出现,蛇要在此范围内运行,显示场地内的所有对象、场地边框、豆和蛇。

//场地类,生成一个画布和豆、蛇
function Farm()
var t = this;
ctx.fillStyle ='white';
ctx.fillRect(0,0,canvas.width,canvas.height);foods =l;//有环莲浮墩甘伴榜机郧扒唉癌挨膊鞍皑傲卑唉扮诚成一个食物t.addfood = function()var x= parseInt(canvas.width / gridWidth * Math.random()) * gridWidth;var y=parseInt(canvas.height / gridWidth * Math.random()) * gridWidth;var food = new Eood(x,y,gridWidth);food把椽斌新foodInit();
//重新初始化豆数组,不要把前一次游戏的数组元素遗留
foods.push(food);
//重新初始化蛇身(块)数组,不要把前一次游戏的数组元素遗留snakes =;//更新速度 500 毫秒(即移动速度)
t.snake= new Snake(100,100,5,500); //初始 5 节长度,位置(100,100)处//*彼啊画钻班等滁癌报蛇t.snake.init();
}

 

5. 主程序

在游戏开始后,要首先初始化场地Farm类,显示场地内的所有对象,场地边框、豆和蛇。同时要2秒随机产生一个新豆并显示。

var canvas = document.getElementById("canvas”)
var ctx = canvas.getContext( 2d');
var gridWidth = 20;
var score = 0;
var foods = new Array(),snakes = new Array();//开始游戏
function gameStart(
var farm = new Farm();
//放豆和蛇的数组
//2 秒产生一个豆
food interval = setInterval(farm.addfood,2000);
gameStart();
//结束
function gameover()var judge = confirm("游戏结束,是否重新开始”);score =0;
textmsg.innerHTML = score +“分clearInterval(snake interval);
//清除产生蛇移动定时
//清除产生新豆定时
clearInterval(food interval);
if(!judge){
//选择不重新开始
return false;
gameStart();
}

至此,贪吃蛇游戏编写完成。 

更多推荐

JS 继承

JS继承的方式一、继承是什么?二、继承实现的方式2.1原型链继承2.2原型式继承2.3构造函数继承2.4组合继承2.5寄生式继承2.6寄生组合式继承2.7extends方法一、继承是什么?继承(inheritance)是面向对象软件技术当中的一个概念。如果一个类别B“继承自”另一个类别A,就把这个B称为“A的子类”,而

软件工程 第一次随堂练习

以下答案是经过人工智能生成,个人理解得出的答案,若有不同见解,请在评论区留言或私信说明下列需求分别属于下面的哪种类型,为什么?A.业务需求B.用户需求C.系统级(功能)需求D.性能需求E.质量需求F.约束G.对外接口H.数据需求I.过程需求J.项目需求K.其他需求(硬件需求、人力需求等)1.经过10天培训的收银员能熟练

页面静态化、Freemarker入门

页面静态化介绍页面的访问量比较大时,就会对数据库造成了很大的访问压力,并且数据库中的数据变化频率并不高。那需要通过什么方法为数据库减压并提高系统运行性能呢?答案就是页面静态化。页面静态化其实就是将原来的动态网页(例如通过ajax请求动态获取数据库中的数据并展示的网页)改为通过静态化技术生成的静态网页,这样用户在访问网页

《IP编址与路由:网络层的关键技术》

前言:在TCP/IP协议栈中,网络层位于第三层,起到了承上启下的关键作用。它不仅负责处理来自数据链路层和传输层的请求,还需确保数据包的正确转发。本文将深入探讨IP编址与路由的相关知识,帮助您更好地理解网络层的重要性和应用。目录IP地址分类:ARP/RARP协议原理:路由器工作原理:IP地址分类:IPv4和IPv6是两种

Vue基础语法【下】

目录一、事件处理器1.事件修饰符.stop.prevent.capture.self.once2.按键修饰符.enter.tab.delete.esc.space.up.down.left.right.ctrl、.alt、.shift、.meta二、表单赋值与取值三、自定义组件1.组件介绍2.局部组件3.全局组件4.组

智慧公厕:改变公共厕所管理与运营的未来

在现代社会中,公共厕所是城市建设的重要组成部分。然而,长期以来,公共厕所管理与运营一直是一个令人头疼的问题。由于各种原因,公共厕所常常陷入管理难、环境差、设备设施陈旧的状态,给人们的生活带来困扰。然而,智慧公厕出现正在改变这一局面。通过更透切的感知、更全面的互连和更深入的智能,智慧公厕为城市的公共厕所管理与运营带来了前

FPGA设计时序约束一、主时钟与生成时钟

​目录一、主时钟create_clock1.1定义1.2约束设置格式1.3Addthisclocktotheexistingclock1.4示例1.5差分信号二、生成时钟generate_clock2.1定义2.2格式2.2.1byclockfrequency2.2.2byclockedges2.2.3示例2.2.4自

python学习之【文件读写】

前言上一篇文章​​python学习——【第十四弹】​​​中学习了python中的包与内置模块,这篇文章接着学习python中的文件读写。编码方式在学习文件读写之前,我们先了解下python当中的编码方式:字节(Byte)是计量单位,表示数据量多少,是计算机信息技术用于计量存储容量的一种计量单位,通常情况下一字节等于八位

FPGA/数字IC(芯海科技2022)面试题 2(解析版)

以下仅为学习参考(非原创),如有疑惑欢迎评论区指出!一、单选题(共20题,每题3分,共60分)1.D触发器:Tsetup=3ns,Thold=1ns,Tck2q=1ns,该D触发器最大可运行时钟频率是()A、1GHZB、500MHZC、250MHZD、200MHZ解:C最大可运行时钟频率与保持时间无关,1/(Tsetu

redis群集

目录redis群集模式主从复制主从复制的作用主从复制流程主从复制模型搭建Redis主从复制哨兵模式哨兵模式原理哨兵结构故障转移机制主节点选举机制搭建Redis哨兵模式群集模式集群的作用集群的数据分片搭建群集模式redis群集模式redis群集有三种模式,分别是主从同步/复制、哨兵模式、Cluster,下面会讲解一下三种

Qt5开发及实例V2.0-第九章-Qt文件及磁盘处理

Qt5开发及实例V2.0-第九章-Qt文件及磁盘处理第9章Qt5文件及磁盘处理9.1读写文本文件9.1.1QFile类读写文本9.1.2QTextStream类读写文本9.2读写二进制文件9.3目录操作与文件系统9.3.1文件大小及路径获取实例9.3.2文件系统浏览9.4获取文件信息9.5监视文件和目录变化本章相关例程

热文推荐