手撕A*(效果不太好)

根据这篇博客基于C++实现的A*算法(链表和二叉堆实现)_a*算法是不是必须用到链表?-CSDN博客修改了A*,用优先级队列和unordered_set,但是效果不太好,时间反而增加了,正在探索原因。

#include<vector>
#include<algorithm>
#include<iostream>
#include<queue>
#include<unordered_set>struct Node {Node(int X, int Y, std::shared_ptr<Node> p = nullptr) : x(X), y(Y), prev(p) {}int x;  //点的x坐标int y;  //点的y坐标int G = 0;  //起点到该点的欧拉距离int H = 0;  //该点到终点的曼哈顿距离int F = 0;  //G+Hstd::shared_ptr<Node> prev;  //指向的前一个节点bool operator<(const Node& other_node) const { return this->F > other_node.F; }
};class AStar {
public:AStar(std::vector<std::vector<int> > m): maps_(m) {}std::shared_ptr<Node> FindPath(std::shared_ptr<Node> begin, std::shared_ptr<Node> end);void PrintAStarPath(const std::pair<int, int> &, const std::pair<int, int> &);~AStar() {open_set_.clear();close_set_.clear();}private:void RefreshOpenList(std::shared_ptr<Node>, std::shared_ptr<Node> end);int CalculateH(std::shared_ptr<Node>, std::shared_ptr<Node>) const;int CalculateF(std::shared_ptr<Node>,std::shared_ptr<Node>) const;private:std::vector<std::vector<int> > maps_;  //地图std::unordered_set<std::shared_ptr<Node>> open_set_;  //保存还未遍历过的节点std::priority_queue<Node> open_queue_;  //使用优先级队列std::unordered_set<std::shared_ptr<Node> > close_set_;  //保存已经找到最短路径的节点const static int cost_low_;  //上下位移的距离const static int cost_high_;  //斜向位移的距离
};const int AStar::cost_low_ = 10;
const int AStar::cost_high_ = 14;
int AStar::CalculateH(std::shared_ptr<Node> point, std::shared_ptr<Node> end) const {return cost_low_ * (std::abs(point->x - end->x) + std::abs(point->y - end->y));
}int AStar::CalculateF(std::shared_ptr<Node> point,std::shared_ptr<Node> end) const {return point->G + CalculateH(point, end);
}std::shared_ptr<Node> AStar::FindPath(std::shared_ptr<Node> begin, std::shared_ptr<Node> end) {open_set_.emplace(begin);open_queue_.emplace(*begin);RefreshOpenList(begin, end);while (!open_set_.empty()) {auto iter = std::make_shared<Node>(open_queue_.top().x, open_queue_.top().y, open_queue_.top().prev);close_set_.emplace(iter);std::shared_ptr<Node> iter_temp = iter;open_set_.erase(iter);open_queue_.pop();RefreshOpenList(iter_temp, end);auto iter2 = std::find_if(open_set_.cbegin(), open_set_.cend(), [end](std::shared_ptr<Node> sp){ return (sp->x == end->x) && (sp->y == end->y); });if (iter2 != open_set_.end())return *iter2;}return nullptr;
}
void AStar::RefreshOpenList(std::shared_ptr<Node> point, std::shared_ptr<Node> end) {bool upIsWall = false;  //表示当前点上有障碍物,即对应的斜向没法走bool downIsWall = false;bool leftIsWall = false;bool rightIsWall = false;if (point->x - 1 >= 0 && maps_[point->x - 1][point->y] == 1)upIsWall = true;if (point->x + 1 < int(maps_.size()) && maps_[point->x + 1][point->y] == 1)downIsWall = true;if (point->y - 1 >= 0 && maps_[point->x][point->y - 1] == 1)leftIsWall = true;if (point->y + 1 < int(maps_.front().size()) && maps_[point->x][point->y + 1] == 1)rightIsWall = true;for (int i = point->x - 1; i <= point->x + 1; ++i) {for (int j = point->y - 1; j <= point->y + 1; ++j) {if (i >= 0 && j >= 0 && i < int(maps_.size()) && j < int(maps_.front().size()) && (i != point->x || j != point->y) && !maps_[i][j]) {if (i != point->x && j != point->y) {if (leftIsWall && j < point->y)continue;if (rightIsWall && j > point->y)continue;if (upIsWall && i < point->x)continue;if (downIsWall && i > point->x)continue;                       }auto cur = std::make_shared<Node>(i, j, point);cur->G = ((i != point->x && j != point->y) ? cost_high_ : cost_low_) + point->G;cur->H = CalculateH(cur, end);cur->F = CalculateF(cur, end);auto iter_close = std::find_if(close_set_.cbegin(), close_set_.cend(), [i,j](std::shared_ptr<Node> sp){ return (sp->x == i) && (sp->y == j); });if (iter_close == close_set_.end()) {auto iter_open = std::find_if(open_set_.cbegin(), open_set_.cend(), [i,j](std::shared_ptr<Node> sp){ return (sp->x == i) && (sp->y == j); });if (iter_open != open_set_.end()) {if((*iter_open)->G > cur->G) {(*iter_open)->G = cur->G;(*iter_open)->F = (*iter_open)->G + (*iter_open)->H;(*iter_open)->prev = point;}}elseopen_set_.emplace(cur); open_queue_.emplace(*cur);}}}}
}void AStar::PrintAStarPath(const std::pair<int, int>& start, const std::pair<int, int>& end) {auto start_sp = std::make_shared<Node>(start.first, start.second), end_sp = std::make_shared<Node>(end.first, end.second);std::shared_ptr<Node> final = FindPath(start_sp, end_sp);if (!final)std::cout << "没有找到起点到终点路径" << std::endl;else {while (final) {maps_[final->x][final->y] = '*';final = final->prev;}for (const auto &i : maps_) {for (const auto &j : i) {if (j > 1)std::cout << char(j) << " ";elsestd::cout << j << " ";}std::cout << std::endl;}}
}int main() {// 记录开始时间  auto start = std::chrono::high_resolution_clock::now();  std::vector<std::vector<int>> map = {{1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1},{1, 0, 0, 1, 1, 0, 1, 0, 0, 0, 0, 1},{1, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 1},{1, 0, 0, 0, 0, 0, 1, 0, 0, 1, 1, 1},{1, 1, 1, 0, 0, 0, 0, 0, 1, 1, 0, 1},{1, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1},{1, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 1},{1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1}};AStar star(map);star.PrintAStarPath({1, 1}, {6, 10});// 记录结束时间  auto end = std::chrono::high_resolution_clock::now();  // 计算运行时间  std::chrono::duration<double> diff = end-start;  std::cout << "Elapsed time: " << diff.count() << " s\n";  return 0;
}

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

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

相关文章

通过dbeaver链接dm8数据库

一、环境说明 windows 11 vmware 17 ubuntu 22 tt:~$ lsb_release -a No LSB modules are available. Distributor ID: Ubuntu Description: Ubuntu 22.04.3 LTS Release: 22.04 Codename: jammytt:~$ docker info Client:Version: 24.0.5Context: d…

数列特征

试题 基础练习 数列特征 提交此题 评测记录 资源限制 内存限制&#xff1a;256.0MB C/C时间限制&#xff1a;1.0s Java时间限制&#xff1a;3.0s Python时间限制&#xff1a;5.0s 问题描述 给出n个数&#xff0c;找出这n个数的最大值&#xff0c;最小值&#xf…

taro常用的一些功能

1.打电话 Taro.makePhoneCall({phoneNumber:that.state.tell,success:function(){}}) 2.弹窗获取位置-->获取本人地理位置 // 弹窗 Taro.showModal({content: 即将前往设置&#xff0c;允许小程序获取您的位置信息,showCancel:false,success: function (res) {if (res.co…

C++之移动语义与智能指针

目录 移动语义 1、几个基本概念的理解 2、复制控制语义的函数 3、移动控制语义的函数 3.1、移动构造函数: 3.2、移动赋值函数 4.区别 5、std::move函数 6.代码演示: 资源管理与智能指针 一、C语言中的问题 二、C的解决办法(RAII技术)&#xff1a; 三、四种智能指针…

【软考】蠕虫病毒

目录 一、概念1.1 说明 二、示例2.1 震网2.2 熊猫烧香2.2 红色代码2.3 爱虫病毒 一、概念 1.1 说明 1.一段可以借助程序自行传播的程序或代码 二、示例 2.1 震网 1.Stuxnet 2.利用系统漏洞破坏工业基础设施&#xff0c;攻击工业控制系统 2.2 熊猫烧香 1.是一种经过多次变种…

2024年产品品牌化深度分析:消费者心理与品牌化、产品质量的权衡

随着市场竞争的加剧和消费者需求的多样化&#xff0c;产品品牌化已经成为企业不可或缺的战略选择。在2024年&#xff0c;当消费者面对众多商品时&#xff0c;品牌化与产品质量之间的权衡成为了消费者决策的重要因素。那么&#xff0c;在消费者心理中&#xff0c;品牌化重要还是…

cadence中run pspice运行仿真 光标搜索Search Command

cadence中run pspice运行仿真 光标搜索Search Command 在cadence进行波形分析时&#xff0c;如果可以随时找到对应的点分析十分方便。 也就是cadence中的光标搜索&#xff08;Search Command&#xff09;功能 但是需要输入正确形式才能使用 官方说明&#xff1a;PSpice User…

深入理解Java反射:原理、机制及应用场景解析

引言 Java反射是一项强大的技术&#xff0c;它允许程序在运行时获取类的信息并操作类或对象的属性、方法及构造方法。本文将深入探讨Java反射的原理、机制&#xff0c;并通过丰富的例子和应用场景来展示其重要性和灵活性。 反射的原理 Java反射的核心原理是通过java.lang.re…

【软件测试_黑白盒测试】白盒测试黑盒测试 区别

从政府工作报告探计算机行业发展 政府工作报告作为政府工作的全面总结和未来规划&#xff0c;不仅反映了国家整体的发展态势&#xff0c;也为各行各业提供了发展的指引和参考。随着信息技术的快速发展&#xff0c;计算机行业已经成为推动经济社会发展的重要引擎之一。因此&…

chatGPT中文在线版本(亲测可用

ChatGPT是一个先进的自然语言处理模型&#xff0c;由OpenAI开发。它通过深度学习技术训练而成&#xff0c;可以进行对话、回答问题等多种自然语言处理任务。对于学生、开发者、研究人员和任何对人工智能感兴趣的人来说&#xff0c;这是一个非常有用的工具。 最近找到一个国内可…

Linux 服务升级:Nginx 热升级 与 平滑回退

目录 一、实验 1.环境 2.Kali Linux 使用nmap扫描CentOS 3.Kali Linux 远程CentOS 4.Kali Linux 使用openvas 扫描 CentOS 5.Nginx 热升级 6.Nginx 平滑回退 二、问题 1.kill命令的信号有哪些 2.平滑升级与回退的信号 一、实验 1.环境 &#xff08;1&#xff09;主机…

鸿蒙网络开发学习:【ylong_http】

简介 ylong_http 构建了完整的 HTTP 能力&#xff0c;支持用户使用 HTTP 能力完成通信场景的需求。 ylong_http 使用 Rust 编写&#xff0c;为 OpenHarmony 的 Rust 能力构筑提供支持。 ylong_http 在 OpenHarmony 中的位置 ylong_http 向 OpenHarmony 系统服务层中的网络协…

Adaptive Object Detection with Dual Multi-Label Prediction

gradient reversal layer (GRL) 辅助信息 作者未提供代码

蓝桥杯需要掌握的几个案例(C/C++)

文章目录 蓝桥杯C/C组的重点主要包括以下几个方面&#xff1a;以下是一些在蓝桥杯C/C组比赛中可能会涉及到的重要案例类型&#xff1a;1. **排序算法案例**&#xff1a;2. **查找算法案例**&#xff1a;3. **数据结构案例**&#xff1a;4. **动态规划案例**&#xff1a;5. **图…

java 高级面试题(借鉴)(下)

雪花算法原理 第1位符号位固定为0&#xff0c;41位时间戳&#xff0c;10位workId&#xff0c;12位序列号&#xff0c;位数可以有不同实现。 优点&#xff1a;每个毫秒值包含的ID值很多&#xff0c;不够可以变动位数来增加&#xff0c;性能佳&#xff08;依赖workId的实现…

数据结构面试题

1、数据结构三要素&#xff1f; 逻辑结构、物理结构、数据运算 2、数组和链表的区别&#xff1f; 数组的特点&#xff1a; 数组是将元素在内存中连续存放&#xff0c;由于每个元素占用内存相同&#xff0c;可以通过下标迅速访问数组中任何元素。数组的插入数据和删除数据效率低…

v77.递归

理解&#xff1a; 函数直接或者间接地调用自身&#xff1b;并且有边界条件。 1&#xff1a; #include <stdio.h> int main() {int result fun(3);printf("%d",result);return 0 ; } int fun(int num) {if(num 1)return num;return num fun(num-1); }思路…

高效的二进制列化格式 MessagePack 详解

目录 MessagePack 序列化原理 MessagePack 数据类型及编码方式 MessagePack 序列化与反序列化过程 MessagePack 的优势 应用场景 注意事项 小结 MessagePack &#xff08;简称 msgPack&#xff09;是一种高效的二进制序列化格式&#xff0c;可以将各种数据类型&#xff…

raise PyAutoGUIException! ! !

在了解pyautogui时&#xff0c;你是否遇到过这样的情况&#xff1a; y pyautogui.locateOnScreen(kk.png) print(y) 在信心满满下输入完成后选择直接运行&#xff0c;结果却是抛出异常的尴尬。 raise PyAutoGUIException( pyautogui.PyAutoGUIException: PyAutoGUI was unable…

JavaScript如何判断一个对象是否为数组?

在JavaScript中&#xff0c;有多种方法可以判断一个对象是否为数组。以下是一些常见的方法&#xff1a; 方法一&#xff1a;使用 Array.isArray() 方法 Array.isArray() 是JavaScript内置的方法&#xff0c;专门用于判断一个对象是否为数组。这是一个非常直观且准确的方法。 …