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

这篇文章是作者人工智能导论课的大作业,发出来供大家学习参考(有完整代码)。想要论文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.mzph.cn/news/672069.shtml

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

相关文章

AR特效自研AI算法技术解决方案

在当今这个高速发展的数字化时代&#xff0c;增强现实&#xff08;AR&#xff09;技术已经成为企业创新和市场竞争的重要手段。美摄科技凭借对AI技术的深厚积累&#xff0c;为企业提供了一套创新的AR特效自研AI算法技术解决方案&#xff0c;旨在满足企业在AR领域的多元化需求。…

选择大语言模型:2024 年开源 LLM 入门指南

作者&#xff1a;来自 Elastic Aditya Tripathi 如果说人工智能在 2023 年起飞&#xff0c;这绝对是轻描淡写的说法。数千种新的人工智能工具被推出&#xff0c;人工智能功能被添加到现有的应用程序中&#xff0c;好莱坞因对这项技术的担忧而戛然而止。 甚至还有一个人工智能工…

代码随想录 Leetcode46. 全排列

题目&#xff1a; 代码&#xff08;首刷自解 2024年2月6日&#xff09;&#xff1a; class Solution { private:vector<vector<int>> res;vector<int> path; public:void backtracking(vector<int>& nums, int depth, vector<bool>& us…

数据结构第十天(排序算法总结)

目录 前言 常数时间复杂度&#xff1a;O(1) 线性时间复杂度&#xff1a;O(n) 线性对数时间复杂度&#xff1a;O(n log n) 平方时间复杂度&#xff1a;O(n^2) 对数时间复杂度&#xff1a;O(log n) 前言 排序算法的学习可以告一段落了。但算法的学习永不停止。 今天&…

redis:七、集群方案(主从复制、哨兵模式、分片集群)和面试模板

redis集群方案 在Redis中提供的集群方案总共有三种&#xff08;一般一个redis节点不超过10G内存&#xff09; 主从复制哨兵模式分片集群 主从复制&#xff08;主从数据同步&#xff09; replid和offset Replication Id&#xff1a;简称replid&#xff0c;是数据集的标记&a…

C++集群聊天服务器 数据模块+业务模块+CMake构建项目 笔记 (上)

跟着施磊老师做C项目&#xff0c;施磊老师_腾讯课堂 (qq.com) 本文在此篇博客的基础上继续实现数据模块和业务模块代码&#xff1a; C集群聊天服务器 网络模块业务模块CMake构建项目 笔记 &#xff08;上&#xff09;-CSDN博客https://blog.csdn.net/weixin_41987016/article…

显示器校准软件:BetterDisplay Pro for Mac v2.0.11激活版下载

BetterDisplay Pro是一款由waydabber开发的Mac平台上的显示器校准软件&#xff0c;可以帮助用户调整显示器的颜色和亮度&#xff0c;以获得更加真实、清晰和舒适的视觉体验。 软件下载&#xff1a; BetterDisplay Pro for Mac v2.0.11激活版下载 以下是BetterDisplay Pro的主要…

【Linux取经路】探寻shell的实现原理

文章目录 一、打印命令行提示符二、读取键盘输入的指令三、指令切割四、普通命令的执行五、内建指令执行5.1 cd指令5.2 export指令5.3 echo指令 六、结语 一、打印命令行提示符 const char* getusername() // 获取用户名 {return getenv("USER"); }const char* geth…

LeetCode-第171题-Excel表的序列号

1.题目描述 给你一个字符串 columnTitle &#xff0c;表示 Excel 表格中的列名称。返回 该列名称对应的列序号 。 例如&#xff1a; A -> 1 B -> 2 C -> 3 ... Z -> 26 AA -> 27 AB -> 28 ... 2.样例描述 3.思路描述 遍历时将每个字母与 A 做减法&…

抖音创作服务平台的自适应,从2560到1024,做的真是棒呀。

不得不服&#xff0c;抖音创作服务平台的电脑页面&#xff0c;自适应效果做的非常好&#xff0c;从2560到1024这个跨度的分辨率都做了很好地处理。 2560 4K屏 1920 2K屏幕 1600 宽屏 1440 中宽屏 1366 笔记本屏 1280 窄屏 1024 超窄屏 768 pad端 这个出现横向滚动条了。 算是用…

虚拟飞控计算机:飞行控制系统验证与优化的利器

01.背景介绍 随着航空技术的飞速发展&#xff0c;飞行控制系统作为飞机的心脏&#xff0c;全面负责监测、调整和维持飞行器的姿态、航向、高度等参数&#xff0c;用以确保飞行的安全和稳定。为了满足这些要求&#xff0c;现代飞控系统通常采用先进的处理器和外设来确保其高效、…

一文简介Maven初级使用

一.概述 Maven是专门用于管理和构建Java项目的工具&#xff0c;它的主要功能有&#xff1a; 提供了一套标准化的项目结构提供了一套标准化的项目构建流程&#xff08;编译&#xff0c;测试&#xff0c;打包&#xff0c;发布&#xff09;提供了一套依赖管理机制 一方面&…

【Java EE】----Bean的作用域和生命周期

1.Bean的作用域 定义&#xff1a;Bean 的作⽤域是指 Bean 在 Spring 整个框架中的某种⾏为模式&#xff0c;⽐如 singleton 单例作⽤域&#xff0c;就 表示 Bean 在整个 Spring 中只有⼀份 &#xff08;产生的原因&#xff0c;Bean 默认情况下是单例状态&#xff08;singleton&…

隧道穿透:端口转发、socket隧道代理

目录 端口转发 lcx工具 Lcx工具正向连接 Lcx工具反向连接 SOCKET隧道代理 socks常见利用场景 Proxifier SocksCap64 Proxychains 端口转发 本篇会和搭建介绍一下端口转发和socket隧道代理的概念和简单演示 lcx工具 lcx工具是一个红队人员在内网渗透测试中最典型的端…

俩种方法解决 VScode中 NPM 脚本消失,NPM 脚本未显示在资源管理器侧栏中

npm脚本是npm包管理器的一个功能&#xff0c;允许开发者在package.json文件中定义一系列命令脚本&#xff0c;用于执行各种开发任务。 今天打开准备运行的时候发现找不到NPM脚本了&#xff0c;左侧的一栏完全没有显示&#xff0c;在网上查阅了很多资料后总结出俩个方法可以用来…

存算一体:架构创新,打破算力极限

1 需求背景 在全球数据量呈指数级暴涨&#xff0c;算力相对于AI运算供不应求的现状下&#xff0c;存算一体技术主要解决了高算力带来的高能耗成本矛盾问题&#xff0c;有望实现降低一个数量级的单位算力能耗&#xff0c;在功耗敏感的百亿级AIoT设备上、高能耗的数据中心、自动驾…

2024.02.06

TCP提供面向有连接的&#xff0c;可靠的数据传输服务&#xff0c;传输过程中&#xff0c;数据无误、数据无丢失、数据无失序、数据无重复 UDP面向无连接的&#xff0c;不保证数据可靠的&#xff0c;尽最大努力传输的协议&#xff0c;数据传输过程中&#xff0c;可能出现数据丢…

大路灯护眼灯有必要吗安全吗?五款优秀品质大路灯推荐

随着人们对健康生活的追求和对眼睛健康的关注&#xff0c;落地护眼灯逐渐成为一种热门的家居用品。落地护眼灯主要通过提供适宜的光照条件来保护眼睛&#xff0c;减少眼睛疲劳和视力损害的发生。那么&#xff0c;落地护眼灯是否有必要&#xff1f;它的使用是否安全呢&#xff1…

外包干了10个月,技术退步明显.......

先说一下自己的情况&#xff0c;大专生&#xff0c;18年通过校招进入武汉某软件公司&#xff0c;干了接近4年的功能测试&#xff0c;今年年初&#xff0c;感觉自己不能够在这样下去了&#xff0c;长时间呆在一个舒适的环境会让一个人堕落! 而我已经在一个企业干了四年的功能测…

三网码支付系统源码,三网免挂有PC软件,有云端源码,附带系统搭建教程

搭建教程 1.先上传云端源码 然后配置Core/Config.php文件里面数据库信息注改&#xff1b;数据库帐号密码 2.云端源码里面Core/Api_Class/Instant_Url_List.php文件配置终端地址注改&#xff1b;第4 http://终端地址/ 3.导入云端数据库 账号admin 密码123456注改&#xff1…