新闻中心
A 算法路径探索中断问题解析与修正

本文深入探讨 a* 寻路算法在实现中可能遇到的一个常见问题:算法在探索少量节点后停止,未能抵达目标。我们将详细分析导致此问题的一个关键编程错误——在邻居节点探索时错误地使用了起始节点而非当前节点,并提供正确的代码示例及实现 a* 算法的关键注意事项,确保算法能够正确高效地找到路径。
A* 算法核心原理概览
A* 算法是一种广泛应用于游戏、机器人路径规划等领域的最佳优先搜索算法,它通过结合 Dijkstra 算法的实际代价(gCost)和贪婪最佳优先搜索的启发式估计代价(hCost),来高效地找到从起点到终点的最短路径。每个节点的总代价 fCost 计算公式为 fCost = gCost + hCost。
A* 算法的核心组成部分包括:
- 开放列表 (Open Set):一个优先队列,存储所有待探索的节点,并根据它们的 fCost 进行排序,fCost 最低的节点优先被探索。
- 关闭列表 (Closed Set):存储所有已探索过的节点,避免重复处理。
- gCost:从起始节点到当前节点的实际移动代价。
- hCost (启发式函数):从当前节点到目标节点的估计移动代价。一个好的启发式函数能够显著提高算法效率。
- cameFrom:一个字典,记录每个节点是通过哪个前驱节点到达的,用于最终路径的回溯。
问题现象与根源分析
在实现 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
查看详情
- 除了 start_node 及其直接邻居之外,其他任何节点都不会被添加到 openSet 中。
- gCost 和 fCost 将无法正确地为远离 start_node 的节点计算和更新。
- openSet 将很快耗尽,因为没有新的、更远的节点被加入,导致算法过早停止。
修正方案
问题的修正非常直接,只需将 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 算法实现,我们将引入一个更适合 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 算法路径探索中断问题解析与修正的详细内容,更多请关注其它相关文章!
# 如何实现
# 国内seo是什么教程
# 福州网站优化软件哪家好
# seo币
# 广州抖音seo收费吗
# 大西瓜seo
# 屏蔽垃圾关键词排名
# 郑州网页优化网站优化
# 宁夏seo优化招商
# 罗源公司推广营销
# 贷款公司网站优化
# 是一个
# 安装包
# node
# 正确地
# 曼哈顿
# 提供一个
# 更高
# 适用于
# 的是
# 如何用
# cos
# 常见问题
# ai
# app
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
如何在Python中使用Optional类型处理可变对象并避免Pylint警告
外媒分析《GTA6》定价:卖100美元可以但真没必要!
抖音极速版最新版本 抖音极速版官方下载地址
Win10怎么设置静态IP地址 Win10手动配置IP地址步骤【指南】
Lar*el DB::listen 事件中的查询执行时间单位解析
必由学官网首页入口 必由学教师网页版登录指南
J*aScript打印功能_j*ascript输出控制
TikTok网页版直接登录 TikTok网页端官方平台入口
c++如何实现一个简单的软件渲染器_c++从零开始的3D图形学
ArrayList与LinkedList操作复杂度详解:遍历与修改
荣耀Play7TPro怎样在信息App置顶客服对话_iPhone荣耀Play7TPro信息App置顶客服对话【优先查看】
C#中解析不规范的HTML为XML 常见的坑与解决办法
CSS条件样式无法按设备触发怎么排查_media条件语句正确设置解决触发问题
Spring Boot嵌入式服务器与J*a EE:功能支持深度解析
Golang如何使用buffered channel提高性能_Golang buffered channel优化技巧
J*aScript Promise链中如何正确终止后续.then执行并处理错误
C++ map遍历方法大全_C++ map迭代器使用总结
谷歌浏览器无痕模式怎么开 Chrome开启无痕浏览设置方法【教程】
wps文字怎么插入目录并自动更新_wps文字如何插入目录并自动更新方法
Animex动漫社网入口地址 Animex动漫社网正版在线入口
Golang如何实现容器化日志收集与分析_Golang容器日志收集分析方法
Golang如何实现Web文件静态资源服务器_Golang静态资源服务器开发与实践
Golang如何测试channel通信行为_Golang channel通信测试与分析方法
腾讯QQ邮箱登录入口_QQ邮箱官方网站使用地址
Python实现多节点属性重叠度分析教程
《刺客信条4:黑旗》重制版新细节曝光:无缝加载 地图更细致!
LINQ to XML为何解析失败? 深入理解C# XDocument的异常处理
手机CPU怎么影响游戏体验_手机CPU对游戏性能的影响分析
地铁跑酷免费秒玩入口链接 地铁跑酷小游戏免费秒玩网站
内存检查:在VS Code中调试C++时的内存视图
解决 Vaadin 8 中大文件音频播放与定位时出现的 IOException
冬*霸灯泡不亮怎么办_浴霸取暖灯一盏不亮的灯座清洁修复法
JUnit5/Mockito:优雅测试内部依赖与异常处理的实践
mcjs网页版流畅运行 mcjs低配电脑畅玩入口
Sublime Text怎么显示空格和制表符_Sublime显示不可见字符设置
c++如何使用TBB库进行任务并行_c++ Intel线程构建模块
漫蛙网页登录入口 漫蛙漫画官方授权网址
Python中高效且防溢出的双曲正弦计算:基于对数空间的优化策略
拼多多赚钱渠道_拼多多收益来源
C++的std::mdspan是什么_C++23中用于操作多维数组的非拥有视图
DLsite中文平台入口 DLsite官网内容在线查看
Word2013如何插入视频和音频媒体_Word2013媒体插入的多媒体支持
不同用户不同价格! 索尼开启账户个性化定价测试
vivo云服务网页版登录 怎么登录vivo云服务网页版
ACG动漫视频网入口 ACG动漫*免费正版观看地址
Golang如何使用new_Go new分配内存机制讲解
mc.js免安装版 mc.js一键畅玩入口
如何使用Rector自动化升级旧代码_通过Composer安装和配置Rector进行代码重构
解决Flask中Quill编辑器内容提交失败及TypeError的指南
小红书商家版怎样在笔记嵌入商品卡路径_小红书商家版在笔记嵌入商品卡路径【挂载教程】


2025-12-04
浏览次数:次
返回列表
已处理
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
# 如果通过当前节点到达