启发式搜索: A*算法

简介: 启发式搜索: A*算法


回顾: 优先队列BFS、最短路

普通BFS:按层扩展

优先队列BFS:每次从队列中取出当前代价最小的状态进行扩展

优先队列BFS的局限性:

一个状态的当前代价最小,只能说明从起始状态到该状态的代价很小,而在未来的搜索中,从该状态到目标状态可能会花费很大的代价。反之亦然。当前代价较大,也许未来代价较小,总代价反而更优。优先队列BFS缺少对未来的预估。

A*算法 – 估价函数

A*算法是一种启发式搜索(Heuristically Search)算法

A*算法的关键是设计一个估价函数:

  • 以任意“状态”为输入,计算出从该状态到目标状态所需代价的估计值
  • 在搜索中,维护一个堆(优先队列),优先选择“当前代价+未来估价”最小的状态进行扩展

估价函数的设计原则:估值必须比实际更优(估计代价≤未来实际代价)

只要保证上述原则,当目标状态第一次从堆中被取出时,就得到了最优解

为什么?

把好状态估差的后果:

本来在最优解搜索路径上的状态被错误地估计了较大的代价,被压在堆中无法取出,从而导致非最优解搜索路径上的状态不断扩展,直至在目标状态上产生错误的答案

把坏状态估好的后果:

只要估价不大于未来实际代价,这个值总会比最优解更早地被取出,从而得到修正。最坏后果无非就是算的状态多了,跑得慢一些。

否决一个正确idea vs 多看一个垃圾idea

A*算法

A*和优先队列BFS的区别就是:考虑优先级的时候有没有加上未来估价

估价越精准(接近但不超过未来实际代价),A*算法越快

估价等于0,就退化为了优先队列BFS

A*算法的关键:开动脑筋,设计优秀的估价函数(必须要乐观估计,但也要尽量精准)

例如:求第K短路,把当前结点到终点的最短路作为估价函数(最短≤K短)

优先选择“当前走过的路径长度+估价函数”最小的状态扩展

实战

773.滑动谜题

https://leetcode.cn/problems/sliding-puzzle/

class Solution {
public:
    int slidingPuzzle(vector<vector<int>>& board) {
        string start = zip(board);
        string target = zip({{1, 2, 3}, {4, 5, 0}});
        //q.push(start);
        q.push({-evaluate(start), start});
        depth[start] = 0;
        while (!q.empty()) {
            string s = q.top().second;
            q.pop();
            int pos = findZeroIndex(s);
            if (pos >= 3) expand(s, pos, pos - 3);
            if (pos <= 2) expand(s, pos, pos + 3);
            if (pos % 3 != 0) expand(s, pos, pos - 1);
            if (pos % 3 != 2) expand(s, pos, pos + 1);
            if (depth.find(target) != depth.end()) return depth[target];
        }
        return -1;
    }
private:
    int evaluate(string &s) {
        int now[6];
        for (int i = 0; i < 6; i++) {
            if (s[i] != '0')
                now[s[i] - '0'] = i;
        }
        int target[6] = {0 ,0 ,1, 2, 3, 4};
        int ans = 0;
        for (int digit = 1; digit <= 5; digit++) {
            int nowx = now[digit] / 3;
            int nowy = now[digit] % 3;
            int targetx = target[digit] / 3;
            int targety = target[digit] % 3;
            ans += abs(nowx - targetx) + abs(nowy - targety);
        }
        return ans;
    }
    void expand(string& s, int pos, int other) {
        string ns = s;
        swap(ns[pos] , ns[other]);
        if (depth.find(ns) != depth.end()) return;
        depth[ns] = depth[s] + 1;
        q.push({ - depth[ns] - evaluate(ns), ns});
        //q.push(ns);
    }
    string zip(const vector<vector<int>>& board) {
        string res;
        for (int i = 0; i < 2; i++) {
            for (int j = 0; j < 3; j++) {
                res += '0' + board[i][j];
            }
        }
        return res;
    }
    int findZeroIndex(string& s) {
        for (int i = 0; i < 6; i++) if (s[i] == '0') return i;
        return -1;
    }
    //queue<string> q;
    priority_queue<pair<int, string>> q;
    unordered_map<string, int> depth;
};

选做题

八数码

https://www.acwing.com/problem/content/847/

https://www.acwing.com/problem/content/181/

推荐一个零声学院免费公开课程,个人觉得老师讲得不错,分享给大家:Linux,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,ZK,流媒体,CDN,P2P,K8S,Docker,TCP/IP,协程,DPDK等技术内容,立即学习

目录
打赏
0
0
0
0
1
分享
相关文章
基于和声搜索优化算法的机器工作调度matlab仿真,输出甘特图
本程序基于和声搜索优化算法(Harmony Search, HS),实现机器工作调度的MATLAB仿真,输出甘特图展示调度结果。算法通过模拟音乐家即兴演奏寻找最佳和声的过程,优化任务在不同机器上的执行顺序,以最小化完成时间和最大化资源利用率为目标。程序适用于MATLAB 2022A版本,运行后无水印。核心参数包括和声记忆大小(HMS)等,适应度函数用于建模优化目标。附带完整代码与运行结果展示。
阿里云 AI 搜索开放平台:从算法到业务——AI 搜索驱动企业智能化升级
本文介绍了阿里云 AI 搜索开放平台的技术的特点及其在各行业的应用。
105 3
算法系列之搜索算法-深度优先搜索DFS
深度优先搜索和广度优先搜索一样,都是对图进行搜索的算法,目的也都是从起点开始搜索,直到到达顶点。深度优先搜索会沿着一条路径不断的往下搜索,直到不能够在继续为止,然后在折返,开始搜索下一条候补路径。
136 62
算法系列之搜索算法-深度优先搜索DFS
|
2月前
|
算法系列之搜索算法-广度优先搜索BFS
广度优先搜索(BFS)是一种非常强大的算法,特别适用于解决最短路径、层次遍历和连通性问题。在面试中,掌握BFS的基本实现和应用场景,能够帮助你高效解决许多与图或树相关的问题。
106 1
算法系列之搜索算法-广度优先搜索BFS
基于SOA海鸥优化算法的三维曲面最高点搜索matlab仿真
本程序基于海鸥优化算法(SOA)进行三维曲面最高点搜索的MATLAB仿真,输出收敛曲线和搜索结果。使用MATLAB2022A版本运行,核心代码实现种群初始化、适应度计算、交叉变异等操作。SOA模拟海鸥觅食行为,通过搜索飞行、跟随飞行和掠食飞行三种策略高效探索解空间,找到全局最优解。
二分搜索:高效的查找算法
【10月更文挑战第29天】通过对二分搜索的深入研究和应用,我们可以不断挖掘其潜力,为各种复杂问题提供高效的解决方案。相信在未来的科技发展中,二分搜索将继续发挥着重要的作用,为我们的生活和工作带来更多的便利和创新。
104 1
基于禁忌搜索算法的VRP问题求解matlab仿真,带GUI界面,可设置参数
该程序基于禁忌搜索算法求解车辆路径问题(VRP),使用MATLAB2022a版本实现,并带有GUI界面。用户可通过界面设置参数并查看结果。禁忌搜索算法通过迭代改进当前解,并利用记忆机制避免陷入局部最优。程序包含初始化、定义邻域结构、设置禁忌列表等步骤,最终输出最优路径和相关数据图表。
实战演练:利用Python的Trie树优化搜索算法,性能飙升不是梦!
在数据密集型应用中,高效搜索算法至关重要。Trie树(前缀树/字典树)通过优化字符串处理和搜索效率成为理想选择。本文通过Python实战演示Trie树构建与应用,显著提升搜索性能。Trie树利用公共前缀减少查询时间,支持快速插入、删除和搜索。以下为简单示例代码,展示如何构建及使用Trie树进行搜索与前缀匹配,适用于自动补全、拼写检查等场景,助力提升应用性能与用户体验。
120 2