A*(A_star)搜索总结
\(A^*(A star)\)搜索总结
标签:算法——搜索
阅读体验:https://zybuluo.com/Junlier/note/1299772
定义
先复制一则定义
\(A^*\)算法在人工智能中是一种典型的启发式搜索算法
启发中的估价是用估价函数表示的:
h(n)=f(n)+g(n)
其中f(n)是节点n的估价函数
g(n)表示实际状态空间中从初始节点到n节点的实际代价
h(n)是从n到目标节点最佳路径的估计代价。
另外定义h'(n)为n到目标节点最佳路径的实际值。
如果h'(n)≥h(n)则如果存在从初始状态走到目标状态的最小代价的解
那么用该估价函数搜索的算法就叫\(A^*\)算法。
有点繁琐,但也看得过去
通俗来讲
\(A^*\)的核心在于上面所讲到的估价函数
他是干什么用的呢
就是我们在搜索的过程中,保证更优的先搜用的
还是有些繁琐对不对,嗯,我也不大会讲啊(没事我会加油)
嘿,认真看下面,我可认真了的啊。。。
如果一个题目要求我们求前K个代价最小的解(只是一个典型,不是所有题目都这样)
假设我们现在有一个状态在\(now\)
已经要记录到答案里面的代价是\(D\)(我喜欢用这个)
我们发现如果爆搜的话状态会是乱的对不对,肯定会使搜索搜到太多
而如果直接把状态按照\(D\)排序的话不能保证答案就会正确(当然,不然就去贪心去)
所以我们引进一个估价函数\(g[状态]\)
当然要求一般是可以预处理出一个状态到答案状态的最优解
回到前面讲到的当前状态\(now\)
如果我们把与\(now\)并列的所有状态按\(D+g[now]\)排序呢?
既不影响答案的正确性,又可以减少坏状态的转移
(因为题目要求是K个最优状态,而这样待决策状态会有序且跑完K个就可以结束,所以会变快)
好吧,还有点蒙对不对,那我们看例题
例题
洛谷P2901 [USACO08MAR]牛慢跑Cow Jogging
好像其他很多\(oj\)都有,但是\(Bzoj\)是权限。。。
题目简述
要求我们求出从起点n到终点1的最短K条路径的长度
(只能从编号大的点往编号小的点走&边有边权)
很裸对吧?
- 预处理估价函数
先跑一遍反向边的\(SPFA\)预处理出每个点到1的最短路作为估价函数
- 直接跑\(A^*\)(这里用\(Bfs\)实现)
从n号点开始\(Bfs\),用堆来代替队列(实现上面所讲的排序)
这时候先到1节点的肯定答案更优(也就是路径更短)
原因很简单吧:估价函数保证答案合法,而排序之后答案有序
搜到K个到达1节点的路径就可以结束,快的飞起。。。
放个代码?
好不容易写一次注释
#include<bits/stdc++.h>
#define lst long long
#define ldb double
#define N 1050
#define M 10050
#define qw ljl[i].to
using namespace std;
const lst Inf=1e15;
int read()
{
int s=0,m=0;char ch=getchar();
while(!isdigit(ch)){if(ch=='-')m=1;ch=getchar();}
while( isdigit(ch))s=(s<<3)+(s<<1)+(ch^48),ch=getchar();
return m?-s:s;
}
int n,m,K,Done;
bool in[N];
lst dis[N];
queue<int> Q;
int hd[N],cnt;
struct EDGE{int to,nxt,v;}ljl[M<<1];
void Add(int p,int q,int o){ljl[++cnt]=(EDGE){q,hd[p],o},hd[p]=cnt;}
void SPFA()
{
for(int i=2;i<=n;++i)dis[i]=Inf;
while(!Q.empty())Q.pop();
Q.push(1),dis[1]=0,in[1]=true;
while(!Q.empty())
{
int now=Q.front();Q.pop(),in[now]=false;
for(int i=hd[now];i;i=ljl[i].nxt)
if(qw>now&&dis[qw]>dis[now]+ljl[i].v)
{
dis[qw]=dis[now]+ljl[i].v;
if(!in[qw])in[qw]=true,Q.push(qw);
}
}
}
//h[i]=g[i]+f[i]---->ans[i]=D+dis[i]
struct NODE{
lst D;int id;
bool operator<(const NODE &X) const
{
return D+dis[id]>X.D+dis[X.id];
}
};priority_queue<NODE> H;
void A_star_Bfs()
{
while(!H.empty())H.pop();
H.push((NODE){0,n});
while(!H.empty())
{
NODE temp=H.top();
int now=temp.id;H.pop();
if(now==1)
{
printf("%lld\n",temp.D);
if(++Done==K)return;continue;
}
for(int i=hd[now];i;i=ljl[i].nxt)
if(qw<now)H.push((NODE){temp.D+ljl[i].v,qw});
}while(Done<K)++Done,puts("-1");
}
int main()
{
n=read(),m=read(),K=read();
for(int i=1;i<=m;++i)
{
int p=read(),q=read(),o=read();
Add(p,q,o),Add(q,p,o);
}
SPFA(),A_star_Bfs();
return 0;
}
/************
1.A*算法在人工智能中是一种典型的启发式搜索算法
启发中的估价是用估价函数表示的:
h(n)=f(n)+g(n)
其中f(n)是节点n的估价函数
g(n)表示实际状态空间中从初始节点到n节点的实际代价
h(n)是从n到目标节点最佳路径的估计代价。
另外定义h'(n)为n到目标节点最佳路径的实际值。
如果h'(n)≥h(n)则如果存在从初始状态走到目标状态的最小代价的解
那么用该估价函数搜索的算法就叫A*算法。
2.第K最短路的算法
我们设源点为s,终点为t,我们设状态f(i)的g(i)为从s走到节点i的实际
距离,h(i)为从节点i到t的最短距离,从而满足A*算法的要求,
当第K次走到f(n-1)时表示此时的g(n-1)为第K最短路长度。
3.这里是kuai的xzy的。。。别怪我。。。
*************/
总结
暂时就将这么多吧
主要是看到网上没有写的那么通俗的\(A^*\)搜索
就想自己总结一下(其实也不通俗。。。)
撤撤撤溜了溜了_______
A*(A_star)搜索总结的更多相关文章
- 【10.9校内练习赛】【搜索】【2-sat】【树链剖分】【A_star k短路】【差分约束+判负环】
在洛谷上复制的题目! P3154 [CQOI2009]循环赛 题目描述 n队伍比赛,每两支队伍比赛一次,平1胜3负0. 给出队伍的最终得分,求多少种可能的分数表. 输入输出格式 输入格式: 第一行包含 ...
- 洛谷 - P2324 - 骑士精神 - A*搜索
为什么估价是16,因为最后一步复原空格可以恢复两个位置,当然设成17.18都可以. #include<bits/stdc++.h> using namespace std; typedef ...
- poj2449(k短路&A_star模板)
题目链接:http://poj.org/problem?id=2449 题意:给出一个有向图,求s到t的第k短路: 思路:k短路模板题,可以用A_star模板过: 单源点最短路径+高级搜索A*;A*算 ...
- 多源第k短路 (ford + 重新定义编号) / 出发点、终点确定的第k短路 (Spfa+ 启发搜索)
第k短路 Description 一天,HighLights实在是闲的不行,他选取了n个地点,n各地点之间共有m条路径,他想找到这m条路径组成的第k短路,你能帮助他嘛? Input 第一行三个正整数, ...
- SQLSERVER走起微信公众帐号已经开通搜狗微信搜索
SQLSERVER走起微信公众帐号已经开通搜狗微信搜索 请打开下面链接 http://weixin.sogou.com/gzh?openid=oIWsFt-hiIb_oYqQHaBMoNwRB2wM ...
- solr_架构案例【京东站内搜索】(附程序源代码)
注意事项:首先要保证部署solr服务的Tomcat容器和检索solr服务中数据的Tomcat容器,它们的端口号不能发生冲突,否则web程序是不可能运行起来的. 一:solr服务的端口号.我这里的sol ...
- SQLServer地址搜索性能优化例子
这是一个很久以前的例子,现在在整理资料时无意发现,就拿出来再改写分享. 1.需求 1.1 基本需求: 根据输入的地址关键字,搜索出完整的地址路径,耗时要控制在几十毫秒内. 1.2 数据库地址表结构和数 ...
- HTML5轻松实现搜索框提示文字点击消失---及placeholder颜色的设置
在做搜索框的时候无意间发现html5的input里有个placeholder属性能轻松实现提示文字点击消失功能,之前还傻傻的在用js来实现类似功能... 示例 <form action=&quo ...
- bzoj1079--记忆化搜索
题目大意:有n个木块排成一行,从左到右依次编号为1~n.你有k种颜色的油漆,其中第i种颜色的油漆足够涂ci个木块.所有油漆刚好足够涂满所有木块,即c1+c2+...+ck=n.相邻两个木块涂相同色显得 ...
随机推荐
- seleniummaster
http://seleniummaster.com/sitecontent/index.php/component/banners/click/6 Step 1: create a Java proj ...
- JS 图片转blob 转base64
//转换为blob有跨域限制 var loadImageToBlob = function (url, callback) { if (!url || !callback) return false; ...
- 对react的redux的研究(一)
action types export const ADD_TODO = 'ADD_TODO' export const TOGGLE_TODO = 'TOGGLE_TODO' export ...
- 8:Spring Boot Shiro记住密码
1,添加Cookie 2,添加安全管理器中 3,配置记住我, 4 ,在登录页面中加我rememberMe复选框 /** * 1.配置Cookie对象 * 记住我的cookie:rememberMe * ...
- mysql ORDER BY语句 语法
mysql ORDER BY语句 语法 作用:用于对结果集进行排序. 语法:顺序:SELECT * from 表名 ORDER BY 排序的字段名 倒序:SELECT * from 表名 ORDER ...
- 超大文件上传方案(ASP.NET)
ASP.NET上传文件用FileUpLoad就可以,但是对文件夹的操作却不能用FileUpLoad来实现. 下面这个示例便是使用ASP.NET来实现上传文件夹并对文件夹进行压缩以及解压. ASP.NE ...
- luogu 4004 Hello world! 分块 + 并查集 + 乱搞
其实呢,我也不理解这道题咋做,等以后有时间再研究研究 #include <bits/stdc++.h> #define ll long long #define maxn 100002 u ...
- [洛谷P3938]:斐波那契(fibonacci)(数学)
题目传送门 题目描述 小$C$养了一些很可爱的兔子.有一天,小$C$突然发现兔子们都是严格按照伟大的数学家斐波那契提出的模型来进行繁衍:一对兔子从出生后第二个月起,每个月刚开始的时候都会产下一对小兔子 ...
- 使用IDEA 创建Maven项目,外加SSM框架
使用idea 新创建项目 然后 新创建 java .resources 文件夹...... 图上是项目结构 java文件夹下的 文件夹 命名规范 com.nf147(组织名)+ oukele(作者) ...
- 高通平台msm8916修改开机logo【原创】
经过两天的奋战终于把开机logo给搞定了啊. 首先修改开机logo要从哪里入手呢?先分析一下源码看看. ---> void display_image_on_screen() { struct ...