寻路算法实例解析:贪吃蛇AI的实现

本文是寻路算法的实际应用篇,以贪吃蛇的实现为例子。

1.首先看下这个在微博上很火的贪吃蛇gif
 


这次我们尝试用代码来模拟下,说不定上面这个图就是计算机搞的。

2.讲贪吃蛇snake AI之前,我们先看下贪吃蛇移动的特点
 


物理上给人的感觉是整个贪吃蛇往右移了一步,在贪吃蛇非常长的情况下给人的感觉移一步要做很多事情。但是在计算机中我们可以简单的考虑贪吃蛇的移动,假设用一个数组来存储所有组成贪吃蛇的格子,那么移动一步,就是把将来的格子插入到这个数组的头部,然后再去掉这个数组的最后一个元素。我们只做两件事情,就完成了一整条蛇的移动!往下看之前,再仔细考虑下移动这个问题。

在说贪吃蛇AI之前,我们要考虑一个问题:怎样保证贪吃蛇永远不死?我们知道无论往那个方向前进一步,尾巴的格子都会空出来,那么追着贪吃蛇的尾巴移动,就能保证贪吃蛇永远不死!

3.寻路算法之A Star

贪吃蛇一般情况下要找一个最短路线去吃苹果,这个时候A star寻路算法就派上用场了,如果你对A Star 算法还不了解,可以先看下这篇文章:《Cocos2d-x 寻路算法之三 A Star》 。这里用A Star 需要特别注意两个问题:1.整个蛇的身体是不可接触的格子的,要排除掉。2.因为贪吃蛇移动是一个动态的过程,所以每走一步,要重新进行寻路,而不是一次寻路完,走完路线,因为尾巴的位置会不断的空出来。

4.单纯使用寻路算法遇到的问题

4.1会进入死胡同
 


黄色的是贪吃蛇的头部,红色是我们要吃的东西,根据寻路算法,黑色的就是最短路线,可以在脑子里脑补下,吃完这个东西,贪吃蛇就挂了!

4.2 找不到路线
 


在贪吃蛇足够长的情况下,苹果可能会在蛇身体包围的圈中,看上图,黄色表示头部,那么蛇就找不到路线了!

4.3 一味的最短路线吃东西,留下太多洞
 


我们看下这张图,红色的表示现在出现的苹果,买手机号码橙色的表示吃完红色的苹果后,苹果可能出现的位置,如果我们简单的用最短路线去吃红色的,即无脑往左走,那么橙色的就在会出现在蛇的包围圈中,将来要吃这个就非常不利,要走很多步。

这个时候比较好的走法是下图:


尽可能的多绕,尽可能地把空白地方填完,而不是以最短路线去吃,要为将来打算。

5. 贪吃蛇的最佳无脑模式

一想到4.3的问题,我就觉得贪吃蛇AI是非常难的一个问题,难道还是设及到“一笔画问题”,那就太难了。不过还好,玩了这么多局的贪吃蛇,我发现一个无脑模式,可以填满整个游戏区域。见下图:
 


就是在最后一行空出来,留做逃生的路线,然后像弹簧一样无脑向右推进,吃完后,从底部绕回最左边,继续这样的策略,直到填满整个游戏区域。不知道读者懂不懂这样的策略。

6.开始讲我的贪吃蛇AI实现了

一定要先理解上面的东西,才好继续往下看。我们知道追着尾巴跑,蛇就不会死,所以我们以最短路线去吃苹果时,要给自己留条后路,策略1.如果吃完苹果还可以找到到自己尾巴路线的话,才去吃苹果。

下面给出的是伪代码:
 

  1. var canFindPath= false;                              //可以找到吃苹果的路线
  2. var canFindTail = false;                             //可以找到自己尾巴的路线
  3. canFindPath = startPathFinding();                    //开始寻找路线
  4.         if(canFindPath){                             //如果可以找到吃苹果的路线
  5.             moveSnake()                              //移动一条看不见的贪吃蛇去吃
  6.             canFindTail = startPathFinding();        //尝试找自己尾巴路线
  7.             if(canFindTail){                         //如果可以找到自己尾巴路线
  8.                 return safePathCell;                 //返回吃苹果路线的第一步
  9.             }
  10.         }
复制代码


策略2.如果找不到吃苹果路线或者吃完苹果后,会发生找不到自己尾巴路线的话,那么就在头部的周围找一个格子,这个格子要满足两个条件,条件1,走完这个格子要能找到到尾巴的路线,条件2,这个格子到苹果的距离是最远的。

策略2中的条件1,我们肯定是能找的到的,因为我们的AI的基础都是基于策略1。

策略2的伪代码如下:
 

  1. var canFindPath= false;                              //可以找到吃苹果的路线
  2. var canFindTail = false;                             //可以找到自己尾巴的路线
  3. canFindPath = startPathFinding();                    //开始寻找路线
  4.         if(canFindPath){                             //如果可以找到吃苹果的路线
  5.             moveSnake()                              //移动一条看不见的贪吃蛇去吃
  6.             canFindTail = startPathFinding();        //尝试找自己尾巴路线
  7.             if(canFindTail){                         //如果可以找到自己尾巴路线
  8.                 return safePathCell;                 //返回吃苹果路线的第一步
  9.             }
  10.         }
  11. if(canFindPath == false || canFindTail == false ){
  12.             return getACellThatIsFarthestToGoal();
  13.    }
复制代码


注意下策略2中的条件2,我们没有用什么高深的算法,我们仅仅是把蛇往一个到苹果最远的格子移动,运行起我们的贪吃蛇,我们发现贪吃蛇AI可以工作了!!这个就有点哲学的味道了,看似在往远离目标的方向上行动,没有想到蛇刚好自己在绕了,绕掉了大部分的身体后,就可以用策略1找到路线了。

看了好几遍自己贪吃蛇AI的移动,不可避免的会有4.3提到的这个问题,后期留下太多洞了,贪吃蛇要追随自己尾巴,移动很久才能吃到一个苹果,我们知道有标题5 提到的无脑模式,我们在想,要不后期就不要以最短路线去吃苹果,而是无脑绕,来吃,这样反而会快些!

怎样才能产生无脑绕的效果呢?我们发现策略2中的条件2会产生这样的效果,在后期,我们就走离苹果最远的格子且这个格子能有到达尾巴的路线,我们发现这样走无意中就会产生这样的无脑绕效果!有点像圆。圆心就是目标,最终还算会不断地接近目标的!

这样算是贪吃蛇游戏的后期呢?我这里就简单如果蛇的长度等于整个游戏格子数量的一半,就算到后期了。

7.最终贪吃蛇AI效果图:
 



8.贪吃蛇AI在线试玩地址

用Cocos2d html5实现的。

  • 3
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
游戏原理: 贪吃蛇是一款经典的游戏,游戏中的玩家需要操控一条蛇去吃食物,同时避免撞到游戏界面的边缘或者自己的身体。每吃到一块食物,蛇的身体就会变长一截,游戏难度也会逐渐加大。当蛇头和蛇身相撞或者撞到边缘时,游戏结束。 代码实现: 下面是一个简单的贪吃蛇游戏实现,使用 Python 的 Pygame 库实现,具体代码如下: ```python import pygame import random # 初始化 Pygame pygame.init() # 定义常量 SCREEN_WIDTH = 480 SCREEN_HEIGHT = 480 CELL_SIZE = 20 CELL_WIDTH = SCREEN_WIDTH // CELL_SIZE CELL_HEIGHT = SCREEN_HEIGHT // CELL_SIZE UP = (0, -1) DOWN = (0, 1) LEFT = (-1, 0) RIGHT = (1, 0) FPS = 10 # 定义颜色 BLACK = (0, 0, 0) WHITE = (255, 255, 255) RED = (255, 0, 0) # 创建游戏窗口 screen = pygame.display.set_mode((SCREEN_WIDTH, SCREEN_HEIGHT)) pygame.display.set_caption('Snake Game') # 定义贪吃蛇类 class Snake: def __init__(self): self.body = [(CELL_WIDTH // 2, CELL_HEIGHT // 2)] self.direction = random.choice([UP, DOWN, LEFT, RIGHT]) # 移动 def move(self): head = self.body[0] x, y = self.direction new_head = (head[0] + x, head[1] + y) self.body.insert(0, new_head) self.body.pop() # 改变方向 def change_direction(self, direction): if direction[0] + self.direction[0] != 0 and direction[1] + self.direction[1] != 0: self.direction = direction # 吃食物 def eat_food(self): head = self.body[0] x, y = self.direction new_head = (head[0] + x, head[1] + y) self.body.insert(0, new_head) # 判断是否撞到自己或者墙壁 def is_dead(self): head = self.body[0] if head[0] < 0 or head[0] >= CELL_WIDTH or head[1] < 0 or head[1] >= CELL_HEIGHT: return True for i in range(1, len(self.body)): if head == self.body[i]: return True return False # 定义食物类 class Food: def __init__(self): self.position = self.generate_position() # 生成位置 def generate_position(self): x = random.randint(0, CELL_WIDTH - 1) y = random.randint(0, CELL_HEIGHT - 1) return (x, y) # 重新生成位置 def regenerate_position(self): self.position = self.generate_position() # 绘制网格 def draw_grid(): for x in range(0, SCREEN_WIDTH, CELL_SIZE): pygame.draw.line(screen, BLACK, (x, 0), (x, SCREEN_HEIGHT)) for y in range(0, SCREEN_HEIGHT, CELL_SIZE): pygame.draw.line(screen, BLACK, (0, y), (SCREEN_WIDTH, y)) # 绘制贪吃蛇 def draw_snake(snake): for cell in snake.body: x, y = cell pygame.draw.rect(screen, WHITE, (x * CELL_SIZE, y * CELL_SIZE, CELL_SIZE, CELL_SIZE)) # 绘制食物 def draw_food(food): x, y = food.position pygame.draw.rect(screen, RED, (x * CELL_SIZE, y * CELL_SIZE, CELL_SIZE, CELL_SIZE)) # 主函数 def main(): # 创建贪吃蛇和食物 snake = Snake() food = Food() # 创建时钟 clock = pygame.time.Clock() # 游戏循环 while True: # 处理事件 for event in pygame.event.get(): if event.type == pygame.QUIT: pygame.quit() exit() elif event.type == pygame.KEYDOWN: if event.key == pygame.K_UP: snake.change_direction(UP) elif event.key == pygame.K_DOWN: snake.change_direction(DOWN) elif event.key == pygame.K_LEFT: snake.change_direction(LEFT) elif event.key == pygame.K_RIGHT: snake.change_direction(RIGHT) # 移动贪吃蛇 snake.move() # 判断是否吃到食物 if snake.body[0] == food.position: snake.eat_food() food.regenerate_position() # 绘制游戏界面 screen.fill(BLACK) draw_grid() draw_snake(snake) draw_food(food) pygame.display.update() # 判断游戏是否结束 if snake.is_dead(): pygame.quit() exit() # 控制游戏帧率 clock.tick(FPS) if __name__ == '__main__': main() ``` 运行代码后,即可开始游戏。通过方向键控制贪吃蛇的移动,吃到食物后贪吃蛇的身体会变长,撞到自己或者墙壁就会游戏结束。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值