返回

使用 Love2D 引擎开发贪吃蛇游戏

AI 摘要
本文介绍了博主最近使用Love2D游戏引擎开发的贪吃蛇游戏,涉及贪吃蛇的移动算法、Lua编程和游戏状态判断等内容。通过演示贪吃蛇移动规律和判断吃食物的方法,以及通过代码展示蛇的移动和游戏状态的判断逻辑。最后给出完整的程序代码和游戏效果展示,同时提供GitHub链接获取源代码。

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

游戏算法

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

贪吃蛇游戏算法演示1
贪吃蛇游戏算法演示1
贪吃蛇游戏算法演示2
贪吃蛇游戏算法演示2
贪吃蛇游戏算法演示3
贪吃蛇游戏算法演示3
贪吃蛇游戏算法演示4
贪吃蛇游戏算法演示4
通过这四张图的演示,我们可以发现这样一个规律:

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

那么完成这样一个工作需要两个步骤:

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)
  --定义字体
  local 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
    local 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(dt)
  --获取元素个数
  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") then
    dir = 2
  end
  if (key == "d") then
    dir = 3
  end
  if (key == "w") then
    dir = 0
  end
  if (key=="s") then
    dir = 1
  end
end

--计算下一个目标点
function getNextPoint()
  --计算下一个目标点
  local 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 文件后就可以运行了,我们来看看最终的效果:

Love2D 贪吃蛇游戏示例1
Love2D 贪吃蛇游戏示例1

Love2D 贪吃蛇游戏示例2
Love2D 贪吃蛇游戏示例2

本文的项目作为开源项目托管在 Github 上,可以通过Github来获取项目源代码。谢谢大家,今天的内容就是这样了。

Built with Hugo v0.126.1
Theme Stack designed by Jimmy
已创作 275 篇文章,共计 1041161 字