
本文深入探讨 a* 寻路算法在实现中可能遇到的一个常见问题:算法在探索少量节点后停止,未能抵达目标。我们将详细分析导致此问题的一个关键编程错误——在邻居节点探索时错误地使用了起始节点而非当前节点,并提供正确的代码示例及实现 a* 算法的关键注意事项,确保算法能够正确高效地找到路径。
A* 算法是一种广泛应用于游戏、机器人路径规划等领域的最佳优先搜索算法,它通过结合 Dijkstra 算法的实际代价(gCost)和贪婪最佳优先搜索的启发式估计代价(hCost),来高效地找到从起点到终点的最短路径。每个节点的总代价 fCost 计算公式为 fCost = gCost + hCost。
A* 算法的核心组成部分包括:
在实现 A* 算法时,一个常见的错误可能导致算法在探索了少数几个节点后便停止,无法到达目标节点。这种现象通常表现为:算法似乎只处理了起始节点及其直接邻居,然后就提前终止,返回无路径或不完整的路径。
分析原始代码,我们可以发现问题根源在于邻居节点的扩展逻辑:
# 原始问题代码片段
# ...
while not openSet.isEmpty():
current = openSet.dequeue()
if current == end_node:
RetracePath(cameFrom, end_node)
# 错误之处:总是探索 start_node 的邻居
for neighbour in find_neighbors(start_node, graph):
tempGCost = gCost[current] + 1
# ... 后续逻辑代码中 for neighbour in find_neighbors(start_node, graph): 这一行是导致问题的关键。A* 算法的核心在于从 openSet 中取出 current 节点后,需要探索的是 current 节点的邻居,而不是始终探索 start_node 的邻居。
如果总是以 start_node 为基准来寻找邻居,那么:
Tunee AI
新一代AI音乐智能体
1104
查看详情
问题的修正非常直接,只需将 find_neighbors 函数的第一个参数从 start_node 改为 current 即可:
# 修正后的代码片段
# ...
while not openSet.isEmpty():
current = openSet.dequeue()
if current == end_node:
return RetracePath(cameFrom, end_node) # 修正:到达目标后应返回路径
# 正确做法:探索当前节点 (current) 的邻居
for neighbour in find_neighbors(current, graph_nodes):
tempGCost = gCost[current] + 1
# ... 后续逻辑通过这一修改,A* 算法将能够正确地从 current 节点向外扩展,逐步探索整个图,直至找到目标节点或确认无路径可达。
为了提供一个更健壮和完整的 A 算法实现,我们将引入一个更适合 A 算法的 PriorityQueue 实现,它能够处理元素的优先级更新,并提供一个基于网格图的示例。
首先,定义一个能够高效处理优先级更新的优先队列:
import heapq
# 辅助类:优先队列
class PriorityQueue:
def __init__(self):
self.elements = [] # 存储 (priority, item) 元组
self.item_map = {} # 用于快速检查元素是否存在及更新优先级 {item: current_priority}
def isEmpty(self):
return len(self.elements) == 0
def enqueue(self, priority, item):
# 如果元素已存在且新优先级更高(代价更低),则更新
# 这里我们只在新的优先级更优时才更新,或者元素不存在时添加
if item not in self.item_map or priority < self.item_map[item]:
heapq.heappush(self.elements, (priority, item))
self.item_map[item] = priority # 记录或更新元素的当前最佳优先级
def dequeue(self):
while self.elements:
priority, item = heapq.heappop(self.elements)
# 检查 item_map,确保我们处理的是最新的、最低优先级的元素
# 如果 item_map 中的优先级更高,说明这个元素是旧的、无效的(已被更优路径更新)
if item in self.item_map and priority == self.item_map[item]:
del self.item_map[item] # 从 map 中移除,表示
已处理
return item
return None # 队列为空或所有元素都已无效
def contains(self, item):
return item in self.item_map
# 启发式函数(曼哈顿距离,适用于四向移动的网格图)
def heuristic(a, b):
(x1, y1) = a
(x2, y2) = b
return abs(x1 - x2) + abs(y1 - y2)
# 路径回溯函数
def RetracePath(cameFrom, end_node):
path = []
current = end_node
while current in cameFrom:
path.append(current)
current = cameFrom[current]
path.append(current) # 添加起始节点
path.reverse()
return path
# 查找邻居函数(适用于网格图,假设 graph_nodes 是一个包含所有可通行坐标的集合)
def find_neighbors(node, graph_nodes):
x, y = node
neighbors = []
possible_neighbors = [
(x + 1, y), (x - 1, y), (x, y + 1), (x, y - 1)
]
for neighbor in possible_neighbors:
if neighbor in graph_nodes: # 检查邻居是否在图中且可通行
neighbors.append(neighbor)
return neighbors
# 修正后的 A* 算法主函数
def AStar_corrected(start_node, end_node, graph_nodes):
openSet = PriorityQueue()
openSet.enqueue(0, start_node) # 初始节点的 fCost 为 0 + h(start, end)
infinity = float("inf")
gCost = {}
fCost = {}
cameFrom = {}
# 初始化所有节点的 gCost 和 fCost 为无穷大
for node in graph_nodes:
gCost[node] = infinity
fCost[node] = infinity
gCost[start_node] = 0
fCost[start_node] = heuristic(start_node, end_node)
while not openSet.isEmpty():
current = openSet.dequeue()
# 如果当前节点是目标节点,则回溯路径并返回
if current == end_node:
return RetracePath(cameFrom, end_node)
# 遍历当前节点的所有邻居
for neighbour in find_neighbors(current, graph_nodes):
# 假设每一步的移动代价为 1
tempGCost = gCost[current] + 1
# 如果通过当前节点到达以上就是A 算法路径探索中断问题解析与修正的详细内容,更多请关注其它相关文章!
相关文章:
现代化 SciPy 一维插值:interp1d 的替代方案与最佳实践
必由学官方登录入口 必由学教师学生账号快速访问
win11怎么查看应用耗电情况 Win11电池设置查看应用能耗排行榜【优化】
PPT平滑切换怎么做 PPT炫酷“平滑”切换动画制作教程【必学】
文心一言怎样用批量生成做多版文案_文心一言用批量生成做多版文案【批量创作】
葱吃多了会怎样 葱吃多了会伤胃吗
b站如何看历史记录_b站观看历史找回方法
钉钉视频会议声音异常如何处理 钉钉会议音频修复技巧
ArchiveofOurOwn小说阅读-ArchiveofOurOwn同人作品访问链接
Win11怎么隐藏桌面图标 Win11一键隐藏所有桌面元素及恢复显示
HTML转PPT成品工具有哪些?HTML网页转PPT成品工具大全
如何优雅地解决Livewire文件上传难题?SpatieLivewireFilepond让一切变得简单
深入理解字体排版:Adobe光学字偶距与CSS字偶距的差异与实现
Win10如何清理注册表垃圾 Win10手动清理无效注册表【技巧】
《主播少女的秘密账号迷宫》首支宣传片
PDF怎么合并PDF并保持格式_PDF合并文件保持排版教程
如何使用J*aScript精确选择并批量修改特定父元素下子链接的样式
漫蛙manwa2最新登录网址_漫蛙manwa2手机网页版入口
Centos/Linux 系统下安装 composer 的完整步骤
海棠电脑版入口_通过电脑访问海棠官网阅读
必由学在线入口 必由学网页版快速登录入口
steam官方入口大全 steam账号注册及操作指南
优酷会员付费后没到账怎么办_优酷会员充值异常及解决方法
TikTok评论显示延迟如何处理 TikTok评论刷新优化方法
漫蛙manwa官网登录界面_漫蛙漫画网页版主站入口
火狐浏览器占用内存高卡顿怎么办 火狐浏览器性能优化设置技巧
J*a应用程序首次运行自动创建文件与目录的最佳实践
ACG动漫手机版官网入口 手机ACG动漫APP在线观看正版
Go语言HTML解析:利用Goquery精准获取指定元素内容
C++20的source_location是什么_C++在编译期获取源码位置信息用于日志和断言
解决J*aScript中重复选择项的确认对话框显示问题
钉钉视频会议画面卡顿如何解决 钉钉会议画面优化方法
在Blazor WebAssembly应用中动态注入客户端特定指标代码的策略
Python:递归比较文件夹内容并找出特定类型文件的差异
Lar*el如何生成PDF或Excel文件_Lar*el文档导出工具与使用教程
如何在低配置电脑上搭建轻量级J*a环境_占用更小的环境选择技巧
Kafka Streams中基于消息头条件过滤消息的实现指南
Win11怎么关闭触摸屏_Windows 11禁用HID符合标准触摸屏
J*a ArrayList索引越界异常:动态构建列数据的高效策略
TikTok搜索不到用户发布内容怎么办 TikTok用户内容搜索优化方法
汽水音乐车机版横屏版7.1 汽水音乐车机版横屏版下载入口
新三国志曹操传110级星符试炼夏侯渊极难攻略
2026春节假期时间安排 2026春节假日查询
抖音隐秘迷城小游戏入口_ 抖音冒险解谜小游戏秒玩
处理动态列数据:J*a ArrayList的正确初始化与字符累加教程
知音漫客官网漫画下载_知音漫客网页版阅读记录
怎样把文件彻底粉碎无法恢复_Windows下安全删除敏感数据【隐私保护】
谷歌浏览器怎么给标签页静音_Chrome标签静音快捷操作
高德地图家和公司地址在哪设置 高德地图通勤路线设置方法【超详细】
Win11文件资源管理器卡顿怎么修 Win11重置资源管理器进程优化响应速度【修复方法】