首先讲讲A*算法吧。众所周知,A*算法就是启发式搜索,基本形式就是这样:f(x)=g(x)+h(x);其中f(x)代表在x点所需要的总代价,而g(x)代表:从源点到x点已经耗费的实际代价,h(x)代表从x到终点需要的估计代价,这个函数是一个估计值.而从x到终点真正需要的代价为h*(x),在整个启发式搜索中我们必须保证h(x)<=h*(x);不然的话会由于对当前的估价值过高,则会引起答案的错误。构建A*的关键在于准确的规划一个h(x)函数,使得接近h*(x),这样的搜索会使得答案又快又准。可以想象h(x)过小会使得解空间过大,这样搜索出来的结果会很准确但是速度太慢,而对h(x)的过高估计,即估计代价太大会使得结果不准确。
这样我们可以理解了BFS的搜索过程,BFS的搜索过程中没有考虑到h(x)的估计代价,也就是说h(x)=0,只考虑g(x)的实际代价。这样根据实际代价来进行搜索,虽然可以说是很恶心的A*,同样地我们可以知道,BFS的解空间确实很大。
第一次写A*,目前只会应用在K短路上。不过也有点感觉了,关键在于h(x)的设计!
描述一下怎样用启发式搜索来解决K短路。
首先我们知道A*的基础公式:f(x)=g(x)+h(x);对h(x)进行设计,根据定义h(x)为当前的x点到目标点t所需要的实际距离。也就是说x->t距离,由于有很多的节点都是到t的距离,为了计算这个估计值,当然必须先算出x->t的最短路径长度。显然x的值很多而t的值只有一个,对每个x去求单源点最短路径当然不划算!于是反过来做,从t点出发到其他点的单源点最短路径,这样吧估价函数h(x)都求出来,注意这样求出来的h(x)=h*(x);
然后就可以对构造完的h(x)开始启发式搜索了。
首先的点当然就是定义头结点了,头结点的已消耗代价为0,估计代价为h[s],下一个点为v;进入队列,开始for循环。每次取出队头的f(x)最小的节点对其他节点进行拓展。对当前节点的拓展次数++,若当前节点的拓展次数超过K,显然不符合要求,则不进行拓展。若对t节点的拓展次数恰好为K,则找到了所需要的。对当前节点的拓展次数即为到当前节点的第几短路。找到需要节点的K短路后,返回g(t)即可,也就是通过K次拓展的实际消耗的长度。
在for循环中的入队情况:当前节点的可拓展所有边,的所有状态都入队,当前节点到拓展节点的实际代价为当前节点的实际代价+两节点之间的边长。下个节点就是拓展节点,估计函数的值则为拓展节点到目标节点的距离h(x);
#define N 1004
const int INF = (1<<30);
int n,m;
int dis[N];//当前点到终点的最短距离
bool vis[N];
struct node{
int v;//下一个点
int dis;//距离
};
struct edge{
int v;
int w;
friend bool operator < (edge a,edge b){//重载为小根堆
return a.w+dis[a.v] > b.w+dis[b.v];
}
};
vector<node> mp[N];//正向邻接表
vector<node> remp[N];//反向邻接表
void init(){
int i;
for(i=0;i<=n;i++){
mp[i].clear();
remp[i].clear();
}
}
//用spfa求点到终点的最短距离
bool SPFA(int s){//s是源点编号
queue<int> qq;
int i;
for(i=1;i<=n;++i){
dis[i] = INF; //将除源点以外的其余点的距离设置为无穷大
vis[i] = 0;
}
dis[s] = 0; //源点的距离为0
vis[s] = 1;
qq.push(s);
int u,v;
while(!qq.empty()){
u = qq.front();
qq.pop();
vis[u] = 0;
for(i=0;i<remp[u].size();i++){
node p = remp[u][i];
if(dis[p.v] > p.dis + dis[u]){
dis[p.v] = p.dis + dis[u];
if(!vis[p.v]){
vis[p.v] = 1;
qq.push(p.v);
}
}
}
}
return true;
}
//A*算法求K短路f(x) = g(x)+h(x)f(x)代表在x点所需要的总代价,
//g(x)代表从源点到x点已经耗费的实际代价,h(x)代表从x到终点需要的估计代价
int Astar(int s,int t,int k){//源点,目标点,k短路
if(s==t)k++;//源点与终点一样时k+1
if(dis[s]==INF)return -1;//不能到达终点
edge n1,n2;
priority_queue<edge> pp;//优先队列,每次取出f最小
int cnt[N];
memset(cnt,0,sizeof(cnt));//计算k短路
n1.v = s;
n1.w = 0;
pp.push(n1);
while(!pp.empty()){
n1 = pp.top();
pp.pop();
cnt[n1.v]++;
int len = n1.w;
if(cnt[n1.v]>k)continue;
if(cnt[t] == k)return len;
for(int i = 0;i<mp[n1.v].size();i++){
n2.v = mp[n1.v][i].v;//与n1相邻的下一点
n2.w = mp[n1.v][i].dis+len;
pp.push(n2);
}
}
return -1;
}
int main(){
while(scanf("%d%d",&n,&m) != -1){
int i,j;
init();
while(m--){
int u,v,w;
scanf("%d%d%d",&u,&v,&w);
node tmp;
tmp.v = v;
tmp.dis = w;
mp[u].push_back(tmp);
tmp.v = u;
remp[v].push_back(tmp);
}
int s,t,k;
scanf("%d%d%d",&s,&t,&k);
SPFA(t);
printf("%d\n",Astar(s,t,k));
}
return 0;
}
2341





