基于A-Star搜索算法的迷宫小游戏的设计

news2025/1/9 21:37:50

这篇文章是作者人工智能导论课的大作业,发出来供大家学习参考(有完整代码)。想要论文WORD文件的可以在本文资源处下载(可能还在审核)。

摘要: 本文章聚焦于基于A-Star搜索算法的迷宫小游戏设计,通过深入剖析A-Star算法的核心理论,涵盖当前代价、预估代价、启发函数等关键概念,同时结合Pygame技术的实际应用,展示了A-Star算法在路径规划中的高效和准确表现。
关键词: A-Star搜索算法,耗散值,启发函数,曼哈顿距离,Pygame

一、相关理论

A-star搜索算法是一种常用于图搜索和路径规划的启发式搜索算法。它结合了Dijkstra算法的最短路径搜索和贪婪最优化搜索的优势,通过引入一个启发式评估函数来加速搜索过程。A-Star搜索算法是在搜索A算法的基础上,对启发函数进行条件限制得来的。

(一)A搜索算法

  1. 为了尽快找到从初始节点到目标节点的一条耗散值比较小的路径,我们希望所选择的节点尽可能在最佳路径上。为了评价一个节点在最佳路径上的可能性,A算法给出了评价函数的定义: f ( n ) = g ( n ) + h ( n ) f(n)=g(n)+h(n) f(n)=g(n)+h(n)
  2. 其中 n n n为待评价的节点, g ( n ) g(n) g(n)为从初始节点 s s s到节点 n n n的最佳路径耗散值的估计值,也叫做当前代价; h ( n ) h(n) h(n)为从节点 n n n到目标节点 t t t的最佳路径耗散值的估计值,也叫做预估代价, h ( n ) h(n) h(n)即为启发函数。 f ( n ) f(n) f(n)为从初始节点 s s s经过节点 n n n到达目标节点 t t t的最佳路径耗散值的估计值。这里的耗散值指的是路径的代价,根据求解问题的不同,耗散值可以是路径的长度或者是需要的时间等。

(二)A-Star搜索算法

  1. 在A搜索算法中,由于没有对启发函数 h ( n ) h(n) h(n)做出任何规定和限制,所以A搜索算法得到的结果并不好评定。在A-Star搜索算法中,对启发函数 h ( n ) h(n) h(n)做出以下规定: h ( n ) < h ∗ ( n ) h(n)<h^*(n) h(n)<h(n)
  2. h ∗ ( n ) h^*(n) h(n)表示从节点 n n n到目标节点 t t t的所有路径中的最小耗散值,即最小代价。启发函数满足以上条件的A搜索算法就称作A-Star搜索算法,简称 A ∗ A^* A算法。 h ∗ ( n ) h^*(n) h(n)为启发函数的上限值,如果启发函数大于这个上限值,搜索算法就会出现发散,就不能保证总能找到最优解。

(三)曼哈顿距离

  1. 曼哈顿距离也称为L1范数,是空间中两点之间的距离度量方式。在二维空间中,曼哈顿距离是两点在水平和垂直方向上的距离总和,而不考虑斜线距离。在三维或更高维空间中,曼哈顿距离是各坐标差的绝对值之和。曼哈顿距离是一种常用的启发函数,对于平面网格地图,曼哈顿距离计算从当前节点到目标节点沿着方格网格的最短路径的步数。
  2. 公式: h ( n ) = ∣ c u r r e n t x − g o a l x ∣ + ∣ c u r r e n t y − g o a l y ∣ h(n)=|current_x-goal_x |+|current_y-goal_y | h(n)=currentxgoalx+currentygoaly

二、结果展示:

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

三、详细代码

这里直接附上完整代码,直接放到pycharm里就能运行(要提前下载pygame包)。

import pygame  
import math  
from queue import PriorityQueue  
import tkinter as tk  
from tkinter import messagebox  
  
# pygame初始化  
pygame.init()  
  
# 设置窗口大小  
WIDTH = 900 # 高度与宽度一致  
WIN = pygame.display.set_mode((WIDTH, WIDTH)) # pygame窗口大小  
pygame.display.set_caption("A-star")  
  
# 定义颜色  
RED = (255, 0, 0) # close表  
GREEN = (0, 255, 0) # open表  
BLUE = (0, 0, 255) # end  
YELLOW = (255, 255, 0) # start  
WHITE = (255, 255, 255)  
BLACK = (0, 0, 0) # 障碍  
PURPLE = (128, 0, 128) # A-star算法下的最短路径  
ORANGE = (255, 165, 0) # Start点  
GREY = (128, 128, 128) # 灰色的网格线  
PINK = (255, 0, 255) # End点  
  
# 提示弹窗  
def show_popup_message(message):  
root = tk.Tk()  
root.withdraw() # 隐藏主窗口  
  
# 弹出窗口  
messagebox.showinfo("A-star最短路径搜索·操作介绍", message)  
  
# 文字内容  
message_content = "1. 第一次点击鼠标左键,放置“Sta”方块\n2. 第二次点击鼠标左键,放置“End”方块\n3. 继续点击鼠标左键,放置“障碍物”方块\n4. 选中方块点击鼠标右键,清除方块\n5. 按下空格开始寻找最短路径\n6. 按下q键清空当前页面"  
  
# 创建方格类  
class Spot:  
def __init__(self, row, col, width, total_rows):  
self.row = row # 行  
self.col = col # 列  
self.x = row * width # 行坐标  
self.y = col * width # 列坐标  
self.color = WHITE  
self.neighbors = [] # 相邻点的列表  
self.width = width  
self.total_rows = total_rows  
self.text = "" # 用于存储文字内容  
  
def get_pos(self):  
return self.row, self.col  
  
def is_closed(self):  
return self.color == RED # close表  
  
def is_open(self):  
return self.color == GREEN # open表  
  
def is_barrier(self):  
return self.color == BLACK # 障碍  
  
def is_start(self):  
return self.color == ORANGE # 起点  
  
def is_end(self):  
return self.color == TURQUOISE # 终点  
  
def make_start(self):  
self.color = ORANGE  
  
def make_closed(self):  
self.color = RED  
  
def make_open(self):  
self.color = GREEN  
  
def make_barrier(self):  
self.color = BLACK  
self.text = "" # 用来清除原先留下的文本  
  
def make_end(self):  
self.color = PINK  
  
def make_path(self): # 回溯路径使用  
self.color = PURPLE  
  
def make_clear(self):# 重置对应方格(重置为WHITE)  
self.color = WHITE  
self.text = "" # 用来清除原先留下的文本  
  
def set_text(self, text): # 标记文本  
self.text = text  
  
def draw(self, win):  
pygame.draw.rect(win, self.color, (self.x, self.y, self.width, self.width))  
if self.text:  
font = pygame.font.Font(None, 24)  
text_surface = font.render(self.text, True, WHITE)  
text_rect = text_surface.get_rect(center=(self.x + self.width // 2, self.y + self.width // 2))  
win.blit(text_surface, text_rect)  
  
def update_neighbors(self, grid):  
self.neighbors = []  
if self.row < self.total_rows - 1 and not grid[self.row + 1][self.col].is_barrier(): # 向下搜索,添加下方格到neighbors  
self.neighbors.append(grid[self.row + 1][self.col])  
  
if self.row > 0 and not grid[self.row - 1][self.col].is_barrier(): # 向上搜索,添加上方格到neighbors  
self.neighbors.append(grid[self.row - 1][self.col])  
  
if self.col < self.total_rows - 1 and not grid[self.row][self.col + 1].is_barrier(): # 向右搜索,添加右方格到neighbors  
self.neighbors.append(grid[self.row][self.col + 1])  
  
if self.col > 0 and not grid[self.row][self.col - 1].is_barrier(): # 向左搜索,添加左方格到neighbors  
self.neighbors.append(grid[self.row][self.col - 1])  
  
def __lt__(self, other):  
return False  
  
# 计算两个点的曼哈顿距离  
def Mh(p1, p2):  
x1, y1 = p1  
x2, y2 = p2  
return abs(x1 - x2) + abs(y1 - y2)  
  
# 构造路径  
def reconstruct_path(came_from, current, draw):  
while current in came_from:  
current = came_from[current]  
current.make_path()  
draw()  
  
# A* 算法实现  
# 参考链接:https://www.redblobgames.com/pathfinding/a-star/introduction.html#graphs(写的很好)  
def A_star_algorithm(draw, grid, start, end):  
count = 0  
open_set = PriorityQueue() # open表,用于存储待探索的节点  
open_set.put((0, count, start)) # 将起点放入优先队列,优先级为0  
came_from = {} # 当前方块到之前方块的映射  
g_score = {spot: float("inf") for row in grid for spot in row} # 当前代价,预估代价为曼哈顿距离  
g_score[start] = 0  
f_score = {spot: float("inf") for row in grid for spot in row} # 总代价  
f_score[start] = Mh(start.get_pos(), end.get_pos())  
  
open_set_hash = {start} # keep trace of the nodes  
  
while not open_set.empty(): # open表非空  
  
current = open_set.get()[2]  
open_set_hash.remove(current) # 移出open表  
  
if current == end:  
reconstruct_path(came_from, end, draw)  
end.make_end()  
start.make_start()  
return True  
  
for neighbor in current.neighbors:  
temp_g_score = g_score[current] + 1  
  
if temp_g_score < g_score[neighbor]:  
came_from[neighbor] = current  
g_score[neighbor] = temp_g_score  
f_score[neighbor] = temp_g_score + Mh(neighbor.get_pos(), end.get_pos())  
if neighbor != start and neighbor != end :  
neighbor.set_text(str(f_score[neighbor])) # 标记总代价  
if neighbor not in open_set_hash:  
count += 1  
open_set.put((f_score[neighbor], count, neighbor))  
open_set_hash.add(neighbor)  
neighbor.make_open()  
  
draw()  
  
if current != start:  
current.make_closed()  
  
return False  
  
# 创建方格网格  
def make_grid(rows, width):  
grid = [] # 创建一个空列表,用于存储方格  
gap = width // rows # 一个单元格的大小  
for i in range(rows):  
grid.append([])  
for j in range(rows):  
spot = Spot(i, j, gap, rows)  
grid[i].append(spot)  
return grid # 返回创建的列表  
  
# 绘制网格线  
def draw_grid(win, rows, width):  
gap = width // rows  
for i in range(rows):  
pygame.draw.line(win, GREY, (0, i * gap), (width, i * gap))  
for j in range(rows):  
pygame.draw.line(win, GREY, (j * gap, 0), (j * gap, width))  
  
# 绘制最优路径  
def draw(win, grid, rows, width):  
win.fill(WHITE)  
  
for row in grid:  
for spot in row:  
spot.draw(win)  
  
draw_grid(win, rows, width)  
pygame.display.update()  
  
# 获取鼠标点击的位置  
def get_clicked_pos(pos, rows, width):  
gap = width // rows  
y, x = pos  
  
row = y // gap  
col = x // gap  
  
return row, col  
  
# 主函数  
def main(win, width):  
ROWS = 25 # 一列中方格的个数  
grid = make_grid(ROWS, width)  
# clear_flag = False # 按下空格清屏  
  
start = None  
end = None  
  
run = True  
started = False  
  
show_popup_message(message_content)  
while run:  
  
draw(win, grid, ROWS, width)  
for event in pygame.event.get():  
if event.type == pygame.QUIT:  
run = False  
  
if started:  
continue  
  
if pygame.mouse.get_pressed()[0]: # 左键  
pos = pygame.mouse.get_pos()  
row, col = get_clicked_pos(pos, ROWS, width)  
spot = grid[row][col]  
if not start and spot != end: # 第一次按下对应的是Star  
start = spot  
start.make_start()  
start.set_text("Sta") # 标记Start点  
  
elif not end and spot != start: # 第二次按下对应的是End  
end = spot  
end.make_end()  
end.set_text("End") # 标记End点  
  
elif spot != end and spot != start: # 之后按下的对应的是障碍  
spot.make_barrier()  
  
elif pygame.mouse.get_pressed()[2]: # 右键  
pos = pygame.mouse.get_pos()  
row, col = get_clicked_pos(pos, ROWS, width)  
spot = grid[row][col]  
spot.make_clear()  
if spot == start:  
start = None  
elif spot == end:  
end = None  
  
if event.type == pygame.KEYDOWN: # 检测按键按下  
if event.key == pygame.K_SPACE and start and end : # 空格键对应开始寻找最短路径  
for row in grid:  
for spot in row:  
if spot != start and spot != end and spot.color != BLACK:  
spot.make_clear()  
  
for row in grid:  
for spot in row:  
spot.update_neighbors(grid)  
  
A_star_algorithm(lambda: draw(win, grid, ROWS, width), grid, start, end)  
  
elif event.key == pygame.K_q: # 按下英文字母q键对应清空屏幕  
start = None  
end = None  
grid = make_grid(ROWS, width)  
pygame.event.clear()  
  
pygame.quit()  
  
# 运行主函数  
main(WIN, WIDTH)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1433760.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

蓝桥杯Web应用开发-CSS 基础语法3(文本属性)

CSS 基础语法-文本属性 专栏持续更新中 文本属性 文本属性用于定义文本的样式&#xff0c;通过文本属性&#xff0c;可以改变文本的颜色、字间距、对齐方式、文本修饰和文本缩进等。常用文本属性如下表所示&#xff1a; 属 性可取值描述line-heightnormal、number、length、…

vulhub中Apache APISIX Dashboard API权限绕过导致RCE(CVE-2021-45232)

Apache APISIX是一个动态、实时、高性能API网关&#xff0c;而Apache APISIX Dashboard是一个配套的前端面板。 Apache APISIX Dashboard 2.10.1版本前存在两个API/apisix/admin/migrate/export和/apisix/admin/migrate/import&#xff0c;他们没有经过droplet框架的权限验证&…

【Java报错】显示错误“Error:java: 程序包org.springframework.boot不存在“

使用idea运行项目&#xff0c;显示错误信息如下&#xff1a; 原因是&#xff1a;idea配置的maven加载不到autoconfigure。 解决方案一&#xff1a; 第6步绕过证书语句如下&#xff1a; -Dmaven.wagon.http.ssl.insecuretrue -Dmaven.wagon.http.ssl.allowalltrue 打开终端&am…

C++一维数组

个人主页&#xff1a;PingdiGuo_guo 收录专栏&#xff1a;C干货专栏 铁汁们大家好呀&#xff0c;我是PingdiGuo_guo&#xff0c;今天我们来学习一下数组&#xff08;一维&#xff09;。 文章目录 1.数组的概念与思想 2.为什么要使用数组 3.数组的特性 4.数组的操作 1.定义…

浅谈QT的几种线程的使用和区别。

简介&#xff1a; 线程是操作系统中的基本执行单元&#xff0c;是一个独立的执行路径。每个线程都有自己的栈空间&#xff0c;用于存储本地变量和函数调用的上下文。多个线程可以在同一进程中并发执行&#xff0c;从而实现并发处理&#xff0c;提高程序的性能和响应能力。 与进…

【Docker】入门到精通(常用命令解读)

一、准备工作 1.配置Docker的yum库 首先要安装一个yum工具 yum install -y yum-utils安装成功后&#xff0c;执行命令&#xff0c;配置Docker的yum源&#xff1a; yum-config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo2.安装Docker 执…

Go语言深度解析:探索 crypto/md5 标准库的强大功能

Go语言深度解析&#xff1a;探索 crypto/md5 标准库的强大功能 引言Go语言和MD5的基础知识MD5算法简介Go语言概述Go中的MD5实现 crypto/md5 库的使用方法基本用法处理大型数据安全注意事项 实际案例分析示例1&#xff1a;文件的MD5校验示例2&#xff1a;网络数据的MD5哈希示例3…

第9章 安全漏洞、威胁和对策(9.11-9.16)

9.11 专用设备 专用设备王国疆域辽阔&#xff0c;而且仍在不断扩张。 专用设备是指为某一特定目的而设计&#xff0c;供某一特定类型机构使用或执行某一特定功能的任何设备。 它们可被看作DCS、物联网、智能设备、端点设备或边缘计算系统的一个类型。 医疗设备、智能汽车、…

154基于matlab的二维元胞自动机模拟森林火灾(生命游戏 )和模拟收费站交通流

基于matlab的二维元胞自动机模拟森林火灾&#xff08;生命游戏 &#xff09;和模拟收费站交通流。全国大学生美国建模竞赛&#xff0c;程序已调通&#xff0c;可直接运行。 154 元细胞自动机 森林起火 收费站交通 (xiaohongshu.com)

大人不华,君子务实|复旦大学-华盛顿大学EMBA C18班优秀学生代表周靖毕业典礼演讲

周靖      Ecarx集团首席财务官      亲爱的Mazzeo院长、殷院长、尊敬的各位老师们、家人们以及各位同学们和毕业生们&#xff1a;      今天&#xff0c;当我们站在开启人生新篇章的分岔路&#xff0c;真是五味杂陈&#xff0c;感慨万千。对我们而言&#xff0c;能…

Day 35 | 贪心 860.柠檬水找零 、 406.根据身高重建队列 、 452. 用最少数量的箭引爆气球

860.柠檬水找零 题目 文章讲解 视频讲解 思路&#xff1a;分别列出三种支付方式对应的找零情况 class Solution {public boolean lemonadeChange(int[] bills) {int five 0, ten 0, twenty 0;for (int i 0; i < bills.length; i) {if (bills[i] 5) {five;} else if…

undefined symbol: avio_protocol_get_class, version LIBAVFORMAT_58

rv1126上进行编译和在虚拟机里面进行交叉编译ffmpeg都不行 解决办法查看 查看安装的ffmpeg链接的文件 ldd ./ffmpeg rootEASY-EAI-NANO:/home/nano/ffmpeg-4.3.6# ldd ffmpeg linux-vdso.so.1 (0xaeebd000)libavdevice.so.58 > /lib/arm-linux-gnueabihf/libavde…

防火墙是什么?聊聊如何轻松缓解应用漏洞

数字经济时代&#xff0c;也是应用爆炸的时代。企业越来越多地使用分布式应用架构构建现代微服务&#xff0c;以适应日益增长的应用使用量并提供更高的性能。与此同时却出现许多热点**&#xff0c;如供应链安全、零日漏洞、数据泄露等。忽视安全防护的企业会面临丢失业务的风险…

npm install express -g报错或一直卡着,亲测可解决

问题描述&#xff1a; 最近学习vue3前端框架&#xff0c;安装Node.js之后&#xff0c;在测试是否可行时&#xff0c;cmd窗口执行了&#xff1a;npm install express -g&#xff0c;发现如下图所示一直卡着不动&#xff0c;最后还报错了&#xff0c;网上找了好久&#xff0c;各…

操作系统基础:文件系统基础【下】

&#x1f308;个人主页&#xff1a;godspeed_lucip &#x1f525; 系列专栏&#xff1a;OS从基础到进阶 ⚔️1 文件的基本操作⚖️1.1 总览⚖️1.2 几种基本操作&#x1f52d;1.2.1 创建文件&#x1f52d;1.2.2 删除文件&#x1f52d;1.2.3 打开文件&#x1f52d;1.2.4 关闭文件…

python算法与数据结构---动态规划

动态规划 记不住过去的人&#xff0c;注定要重蹈覆辙。 定义 对于一个模型为n的问题&#xff0c;将其分解为k个规模较小的子问题&#xff08;阶段&#xff09;&#xff0c;按顺序求解子问题&#xff0c;前一子问题的解&#xff0c;为后一子问题提供有用的信息。在求解任一子…

Multisim14.0仿真(四十一)交通信号灯仿真设计

一、功能简介&#xff1a; 1&#xff09;、采用两片74LS192做减法计数器&#xff0c;实现倒计时功能。 2&#xff09;、采用DCD数码管显示时间。 3&#xff09;、采用4个TRAFFIC_LIGHT_SINGLE红绿灯 4&#xff09;、采用74LS160和74LS138实现对红绿灯的逻辑控制。 5&#xff09…

Python 潮流周刊#38:Django + Next.js 构建全栈项目

△△请给“Python猫”加星标 &#xff0c;以免错过文章推送 你好&#xff0c;我是猫哥。这里每周分享优质的 Python、AI 及通用技术内容&#xff0c;大部分为英文。本周刊开源&#xff0c;欢迎投稿[1]。另有电报频道[2]作为副刊&#xff0c;补充发布更加丰富的资讯&#xff0c;…

elementUI 表格中如何合并动态数据的单元格

elementUI 表格中如何合并动态数据的单元格 ui中提供的案例是固定写法无法满足 实际开发需求 下面进行改造如下 准备数据如下 //在表格中 设置单元格的方法 :span-method"spanMethodFun" <el-table :data"tableData" border :span-method"spa…

私有化部署一个吃豆人小游戏

目录 效果 安装步骤 1.安装并启动httpd 2.下载代码 3.启动httpd 使用 效果 安装步骤 1.安装并启动httpd yum -y install httpd 2.下载代码 进入目录 cd /var/www/html/ 下载 git clone https://gitee.com/WangZhe168_admin/pacman-canvas.git 3.启动httpd syste…