今天来介绍博主近期捣腾的一个小游戏[贪吃蛇],贪吃蛇这个游戏相信大家都不会感到陌生吧。今天博主将通过Love2D这款游戏引擎来为大家实现一个简单的贪吃蛇游戏,在本篇文章其中我们将会涉及到贪吃蛇的基本算法、Lua语言编程等主要的内容,希望能够对大家开发相似的游戏提供借鉴和思考,文章中如有不足之处,还希望大家能够谅解,由于博主的游戏开发基本就是这样慢慢摸索着学习。所以难免会有不足的地方。

游戏算法

我们首先来看看贪吃蛇是怎么移动的?

通过这四张图的演示,我们能够发现这样一个规律:

蛇的移动事实上是将蛇身体的最后一个元素移动到第一个元素的位置

那么完毕这样一个工作须要两个步骤:

1、将在蛇头位置插入一个新的元素

2、移除蛇尾位置的最后一个元素

好了。了解了蛇的移动后我们再来考虑一个问题,如何推断蛇吃到了食物?思路和蛇的移动相似,主要考虑在蛇头插入的这个元素和食物的关系,假设这个元素的坐标和食物的坐标是同样的,那么就能够觉得蛇吃到了食物,此时蛇的身体应该是变长的。所以仅仅要在蛇头位置插入一个元素就能够了。反之,假设蛇没有吃到食物。那么蛇应该是移动的。所以就能够依照移动的方法来处理了。那么在蛇头位置插入的这个元素该如何确定呢?我们来看以下这段程序:

--计算下一个目标点
function getNextPoint()
--计算下一个目标点
snake={}
if(dir==0) then
snake.x=snakes[1].x
snake.y=snakes[1].y-20
end
if(dir==1) then
snake.x=snakes[1].x
snake.y=snakes[1].y+20
end
if(dir==2) then
snake.x=snakes[1].x-20
snake.y=snakes[1].y
end
if(dir==3) then
snake.x=snakes[1].x+20
snake.y=snakes[1].y
end return snake
end

这里定义了getNextPoint()的方法,目的是计算在蛇头位置加入的下一个元素,这里我们注意到依据蛇的移动方向(dir)的不同,其中0表示上、1表示下、2表示左、3表示右,计算出下一个元素的位置,由于在这个游戏中网格大小是20,所以这里能够直接依据坐标来计算一个元素的位置。

snakes是一个table,保存的是当前的蛇的所有元素的坐标。通过维护这个table,我们就能够利用画图的函数绘制出蛇的身体,这样蛇就能够移动起来了。

我们来看看蛇是如何移动的:

--核心算法——蛇的移动
function SnakeUpdate()
--获取元素个数
local n=table.maxn(snakes)
if(table.maxn(snakes)>0) then
if(getNextPoint().x==foodX and getNextPoint().y==foodY) then
--将下一个目标点的位置插入表中
table.insert(snakes, 1, getNextPoint())
--将食物状态设置为BeEated
foodState="BeEated"
else
--将下一个目标点的位置插入表中
table.insert(snakes, 1, getNextPoint())
--移除最后一个元素
table.remove(snakes,n+1)
end
end
end

在这里我们定义了一个foodState变量以保存食物的状态。当食物的状态为BeEated的时候表示食物被蛇吃掉了,此时应该又一次生成一个食物的坐标。此时事物的状态将变成WaitToEat。食物的坐标保存在foodX和foodY这两个变量中。大家能够到完整的代码中去查看。

我们知道蛇碰到四周墙壁的时候就会死亡,此时游戏结束。

这个比較简单,仅仅要推断蛇头的坐标和屏幕的关系就能够了。

由于在这个游戏中屏幕的尺寸为640X640。所以推断游戏是否结束的代码能够这样写:

--推断游戏状态
if(snakes[1].x<=0 or snakes[1].x>=640 or snakes[1].y<=0 or snakes[1].y>=640) then
gameState=0
else
gameState=1
end

这里gameState为0表示游戏结束。gameState为1表示游戏正常进行。

在完毕了这些核心的算法以后,剩下的事情就交给Love2D引擎来绘制吧。最后给出完整的程序代码:

--定义窗体宽度和高度
local w=640
local h=640
--定义网格单元大小
local unitSize=20; --方块的初始位置
local initX=320
local initY=320 --移动方向
local dir=1 --贪吃蛇集合
local snakes={} --食物状态
--WaitToEat:绘制食物
--BeEated:随机生成食物
local foodState="WaitToEat" --游戏状态
--0:游戏结束
--1:游戏正常
local gameState=1 --食物的位置
local foodX=0
local foodY=0 --Love2D载入事件
function love.load()
--设置窗体标题
love.window.setTitle("Love2D-贪吃蛇游戏")
--设置窗体大小
love.window.setMode(w,h)
--定义字体
myFont=love.graphics.newFont(30)
--设置字体
love.graphics.setFont(myFont)
--设置背景色
love.graphics.setBackgroundColor(255,255,255,255)
--设置线条类型为平滑
love.graphics.setLineStyle("smooth")
--设置线宽
love.graphics.setLineWidth(0.1) --蛇的初始化(蛇的长度为5)
for i=1,5 do
snake={}
snake.x=initX +(i-1) * 20
snake.y=initY
snakes[i]=snake
end --食物初始化
foodX=love.math.random(32-1)*20
foodY=love.math.random(32-1)*20
end --Love2D绘制事件
function love.draw()
--绘制竖线
love.graphics.setColor(0,0,0,255)
for i=0,w,unitSize do
love.graphics.line(0,i,h,i)
end
--绘制横线
for j=0,h,unitSize do
love.graphics.line(j,0,j,w)
end --绘制蛇
for i=1,table.maxn(snakes) do
love.graphics.setColor(0,0,255,255)
love.graphics.rectangle("fill",snakes[i].x,snakes[i].y,20,20)
end --绘制食物
if(foodState=="WaitToEat") then
love.graphics.setColor(255,0,0,255)
love.graphics.rectangle("fill",foodX,foodY,20,20)
end --假设游戏结束则显示GameOver
if(gameState==0) then
love.graphics.setColor(255,0,0,255)
love.graphics.print("Game Over",250,300)
end
end --
function love.update(dt)
--推断游戏状态
if(snakes[1].x<=0 or snakes[1].x>=640 or snakes[1].y<=0 or snakes[1].y>=640) then
gameState=0
else
gameState=1
end --假设游戏状态为正常
if(gameState==1) then
SnakeUpdate()
FoodUpdate()
end
end --核心算法——蛇的移动
function SnakeUpdate()
--获取元素个数
local n=table.maxn(snakes)
if(table.maxn(snakes)>0) then
if(getNextPoint().x==foodX and getNextPoint().y==foodY) then
--将下一个目标点的位置插入表中
table.insert(snakes, 1, getNextPoint())
--将食物状态设置为BeEated
foodState="BeEated"
else
--将下一个目标点的位置插入表中
table.insert(snakes, 1, getNextPoint())
--移除最后一个元素
table.remove(snakes,n+1)
end
end
end --随机生成食物
function FoodUpdate()
--假设食物被蛇吃掉则又一次生成食物
if(foodState=="BeEated") then
foodX=love.math.random(32-1)*20
foodY=love.math.random(32-1)*20
foodState="WaitToEat"
end
end --依据玩家按下的键位定义不同的方向
function love.keypressed(key)
if(key=="a" and dir~=3) then
dir=2
end
if(key=="d" and dir~=2) then
dir=3
end
if(key=="w" and dir~=1) then
dir=0
end
if(key=="s" and dir~=0) then
dir=1
end
end function getNextPoint()
--计算下一个目标点
snake={}
if(dir==0) then
snake.x=snakes[1].x
snake.y=snakes[1].y-20
end
if(dir==1) then
snake.x=snakes[1].x
snake.y=snakes[1].y+20
end
if(dir==2) then
snake.x=snakes[1].x-20
snake.y=snakes[1].y
end
if(dir==3) then
snake.x=snakes[1].x+20
snake.y=snakes[1].y
end return snake
end

将代码压缩成.love文件后就能够执行了,我们来看看终于的效果:



Github

使用Love2D引擎开发贪吃蛇游戏的更多相关文章

  1. Love2D游戏引擎制作贪吃蛇游戏

    代码地址如下:http://www.demodashi.com/demo/15051.html Love2D游戏引擎制作贪吃蛇游戏 内附有linux下的makefile,windows下的生成方法请查 ...

  2. 用Java开发贪吃蛇游戏

    贪吃蛇游戏的设计步骤: Part 1: 设计游戏图纸 画出900*700的白色窗口 在窗口上添加画布 在画布上添加标题 在画布上添加黑色游戏区 Part 2: 放置静态的蛇:一个头.两个身体 加上开始 ...

  3. Android快乐贪吃蛇游戏实战项目开发教程-01项目概述与目录

    一.项目简介 贪吃蛇是一个很经典的游戏,也很适合用来学习.本教程将和大家一起做一个Android版的贪吃蛇游戏. 我已经将做好的案例上传到了应用宝,无病毒.无广告,大家可以放心下载下来把玩一下.应用宝 ...

  4. WebGL实现HTML5的3D贪吃蛇游戏

    js1k.com收集了小于1k的javascript小例子,里面有很多很炫很酷的游戏和特效,今年规则又增加了新花样,传统的classic类型基础上又增加了WebGL类型,以及允许增加到2K的++类型, ...

  5. 100行JS实现HTML5的3D贪吃蛇游戏

    js1k.com收集了小于1k的javascript小例子,里面有很多很炫很酷的游戏和特效,今年规则又增加了新花样,传统的classic类型基础上又增加了WebGL类型,以及允许增加到2K的++类型, ...

  6. H5实现的可自定义贪吃蛇游戏

    原创游戏,使用lufylegend.js开发 用canvas实现的贪吃蛇游戏,与一般的贪吃蛇游戏不同,图片经过美工设计,代码设计支持扩展和自定义. 游戏元素丰富,包括障碍物(仙人掌),金币(奖励),苹 ...

  7. Qt 学习之路 2(32):贪吃蛇游戏(2)

    Qt 学习之路 2(32):贪吃蛇游戏(2) 豆子 2012年12月27日 Qt 学习之路 2 55条评论 下面我们继续上一章的内容.在上一章中,我们已经完成了地图的设计,当然是相当简单的.在我们的游 ...

  8. Qt 学习之路 2(31):贪吃蛇游戏(1)

    Qt 学习之路 2(31):贪吃蛇游戏(1) 豆子 2012年12月18日 Qt 学习之路 2 41条评论 经过前面一段时间的学习,我们已经了解到有关 Qt 相当多的知识.现在,我们将把前面所讲过的知 ...

  9. 基于React的贪吃蛇游戏的设计与实现

    代码地址如下:http://www.demodashi.com/demo/11818.html 贪吃蛇小游戏(第二版) 一年半前层用react写过贪吃蛇小游戏https://github.com/ca ...

随机推荐

  1. 【ASP】在特定的范围内产生N个不同的随机数

    ASP产生一个随机数不难.就两条特定语句: <% Randomize x=int(20*rnd+1) %> 以上的两条语句.表示从1~20这个范围内产生随机数,而且这些随机数都是整数. 那 ...

  2. Android开发之控制手机音频

    本实例通过MediaPlayer播放一首音乐并通过AudioManager控制手机音频.关于AudioManager的具体解释可參照:Android开发之AudioManager(音频管理器)具体解释 ...

  3. 【POJ 2942】Knights of the Round Table(双联通分量+染色判奇环)

    [POJ 2942]Knights of the Round Table(双联通分量+染色判奇环) Time Limit: 7000MS   Memory Limit: 65536K Total Su ...

  4. Visual Studio2013下Magick++配置方法

    声明:本文系作者原创,如需转载请保持文章完整并注明出处(http://blog.csdn.net/u010281174/article/details/52224829). ImageMagick是一 ...

  5. Weex学习与实践(一):Weex,你需要知道的事

    Weex学习与实践(一):Weex,你需要知道的事 http://coderyi.com/posts/weex1/ 1.命令行工具:weex-toolkit  https://github.com/w ...

  6. checkbox改写

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  7. form&method【POST~GET】

    <form.../>中method属性指定了该表单是以哪种方式提交请求,有两种方式:GET请求方式和POST请求方式,默认是GET请求方式.两种方式的区别:get方式的请求是在浏览器地址栏 ...

  8. win32应用禁止改变窗口大小方法

    一种简单的处理方法是在调用CreateWindow函数时指定的窗口样式中去掉WS_THICKFRAME样式. 如果你使用的样式中已经包含该样式,例如WS_OVERLAPPEDWINDOW,我们可以將W ...

  9. 算法 之 aabb

    题目描述:输出所有形如aabb的4位完全平方数(即前两位数字相等,后两位数字也相等). 分支和循环结合在一起时功能强大: 下面列举所有可能的结果aabb,然后判断它们是否为完全平方数.注意a的范围是1 ...

  10. 转:Hibernate中Criteria和DetachedCriteria的完整用法

    原文地址:http://blog.sina.com.cn/s/blog_667528fd0100rkrf.html 设计上可以灵活的根据 Criteria 的特点来方便地进行查询条件的组装.现在对 H ...