【k短路】POJ 2449

首先讲讲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;
}





  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值