最短路与次短路

1、poj3255  求次短路

思路:遍历每条边<u, v>, 看begin[u] + dis<u, v> + end[v] 是否为次短路

//求次短路
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <vector>
#include <cmath>
#include <queue>
#include <stack>
#include <map>
#include <set>

using namespace std;

#define clr(x) memset(x,0,sizeof(x))
#define fp1 freopen("poj3255.txt","r",stdin)
#define fp2 freopen("out.txt","w",stdout)
#define pb push_back

#define INF 0x3c3c3c3c
const int maxn = 5005;

int ss[maxn], tt[maxn];

typedef long long LL;
struct Edge {int from,to,dist;};
struct Node
{
    int d,u;
    bool operator <(const Node &a) const {
        return a.d<d;//从小到大排序。
    }
};

int n,m; //点数和边数,用n表示,e不能和m冲突
vector<Edge> edges;//边列表
vector<int> G[maxn];//每个结点出发的边编号(从0开始编号)
bool done[maxn];//是否已永久编号
int d[maxn];//s到各个点的距离
int p[maxn];//最短路中的上一条边
void init()
{
    for(int i=0;i<n;i++) G[i].clear();//清空邻接表
    edges.clear();
    clr(done);
}

void addedge(int from,int to,int dist)
//如果是无向,每条无向边需调用两次addedge
{
    edges.push_back((Edge){from,to,dist});
    int temp=edges.size();
    G[from].push_back(temp-1);
}

void dijk(int s)
{
    priority_queue<Node> q;
    for(int i=0;i<n;i++) d[i]=INF;
    d[s]=0;
    memset(done,0,sizeof(done));
    q.push((Node){0,s});
    while(!q.empty()) {
        Node x=q.top();
        q.pop();
        int u=x.u;
        if(done[u]) continue;
        done[u]=true;
        for(int i=0;i<G[u].size();i++) {
            Edge &e=edges[G[u][i]];
            if(d[e.to]>d[u]+e.dist) {
                d[e.to]=d[u]+e.dist;
                p[e.to]=G[u][i];
                q.push((Node){d[e.to],e.to});
            }
        }
    }
}

int main()
{
    //fp1;
    int u, v, w;
    int T;
    //scanf("%d", &T);  这地方输入改一下...
    while(T--) {
        scanf("%d %d", &n, &m);{
            init();
            for(int i = 1;i <= m;i++){
                scanf("%d %d %d", &u, &v, &w);
                //printf("%d %d %d\n", u, v, w);
                u--; v--;
                addedge(u, v, w);
                //addedge(v, u, w);
            }
            dijk(0);
            for(int i = 0;i < n;i++) ss[i] = d[i];
            int Min1 = INF;

            edges.clear();
            clr(done);
            dijk(n-1);
            for(int i = 0;i < n;i++) tt[i] = d[i];
            for(int i = 0;i < n;i++){
                for(int j = 0;j < G[i].size();j++){
                    //...存储下来次小边
                    Edge &e=edges[G[i][j]];
                    int temp = ss[i] + e.dist + tt[e.to];
                    //printf("%d~~\n", temp);
                    if(temp>ss[n-1] && temp<Min1){
                        Min1 = temp;
                    }
                }
            }
        }
        printf("%d\n", Min1);
    }
    return 0;
}

2、对于单向边来说(图为单向边),上面(1)那个方法就不适用了。

真正理解Dijkstra ?? ( 感觉还是蛮神的,可以把次短路也按最短路求法那样迭代找出

#include<iostream>
using namespace std;
#define INF 1000100101
#define maxn 1010
#define maxm 11000
struct Edge{
       int v,val,next;
}e[maxm];
int box[maxn];
int n,m,S,T;
int dis[maxn][2],dp[maxn][2],used[maxn][2];
int dij()
{
       int i,j;
       memset(used,0,sizeof(used));
       memset(dp,0,sizeof(dp));
       for(i=1;i<=n;i++)
              dis[i][0]=dis[i][1]=INF;
       dis[S][0]=0;
       dp[S][0]=1;
       for(i=1;i<n*2;i++) {
              int mn=INF,x,flag;
              for(j=1;j<=n;j++) {
                     if(!used[j][0] && mn>dis[j][0])
                     {
                            mn=dis[j][0];
                            x=j;
                            flag=0;
                     }
                     else if(!used[j][1] && mn>dis[j][1])
                     {
                            mn=dis[j][1];
                            x=j;
                            flag=1;
                     }
              }
              if(mn==INF) break;
              used[x][flag]=1;
              int y,val;
              for(j=box[x];j!=-1;j=e[j].next)
              {
                     y=e[j].v;
                     val=e[j].val;
                     if(mn+val<dis[y][0])
                     {
                            dis[y][1]=dis[y][0];
                            dp[y][1]=dp[y][0];
                            dis[y][0]=mn+val;
                            dp[y][0]=dp[x][flag];
                     }
                     else if(mn+val==dis[y][0])
                     {
                            dp[y][0]+=dp[x][flag];
                     }
                     else if(mn+val<dis[y][1])
                     {
                            dis[y][1]=mn+val;
                            dp[y][1]=dp[x][flag];
                     }
                     else if(mn+val==dis[y][1])
                     {
                            dp[y][1]+=dp[x][flag];
                     }
              }
       }
       if(dis[T][0]+1==dis[T][1])
              return dp[T][0]+dp[T][1];
       return dp[T][0];
}
int main()
{
       int i,j,x,y,val,cas;
       scanf("%d",&cas);
       while(cas--)
       {
              scanf("%d%d",&n,&m);
              memset(box,-1,sizeof(box));
              for(i=1;i<=m;i++)
              {
                     scanf("%d%d%d",&x,&y,&val);
                     e[i].v=y;
                     e[i].val=val;
                     e[i].next=box[x];
                     box[x]=i;
              }
              scanf("%d%d",&S,&T);
              printf("%d/n",dij());
       }
       return 0;
}

最短路与次短路,布布扣,bubuko.com

时间: 2024-10-11 06:47:44

最短路与次短路的相关文章

UESTC30-最短路-Floyd最短路、spfa+链式前向星建图

最短路 Time Limit: 3000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) 在每年的校赛里,所有进入决赛的同学都会获得一件很漂亮的T-shirt.但是每当我们的工作人员把上百件的衣服从商店运回到赛场的时候,却是非常累的!所以现在他们想要寻找最短的从商店到赛场的路线,你可以帮助他们吗? Input 输入包括多组数据. 每组数据第一行是两个整数NN ,MM (N≤100N≤100 ,M≤10000M≤1000

hdu1688(dijkstra求最短路和次短路)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1688 题意:第k短路,这里要求的是第1短路(即最短路),第2短路(即次短路),以及路径条数,最后如果最短路和次短路长度差1,则输出两种路径条数之和,否则只输出最短路条数. 思路:dijkstra变形,注意状态的转移,代码上附了注释,就不多说了.. 代码: 1 #include <bits/stdc++.h> 2 #define MAXN 1010 3 using namespace std; 4

hdu 1688 Sightseeing【最短路,次短路条数】

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1688 题意:求最短路和次短路条数如果次短路长度=最短路长度+1 这输出次短路条数+最短路条数,否则输出最短路条数 分析:这是到模版题,献上模版: #include<stdio.h> #include<string.h> #include<algorithm> #include<iostream> #include<queue> #include<

次短路 + 第K短路 模版

虽然从字面上看,次短路和第2短路是一样的.但是我在题目中遇到的却不是这样的. 在有些题目中,需要判断次短路是否存在.比如说,u.v之间只有一条路径.那么只有最短路.次短路是不存在的.这时候,解题方法是先求出最短路,然后枚举删除最短路径中的边,然后求最小值.题目可以看poj3986. 第K短路的实现是 SPFA + A* 算法. A*算法通过一个估价函数f(h)来估计途中的当前点p到终点的距离,并由此决定它的搜索方向,当这条路径失败时,它会尝试其他路径.对于A*,估价函数 = 当前值 + 当前位置

poj3463 Sightseeing --- dij最短路和次短路

最短路好题啊. 题目给定起点和终点,要求最短路和次短路(要求次短路只比最短路大1)的道路数量. 重点在于次短路如何处理是最高效的呢 这就要求对dij算法路径更新的理解了. 我们用一个数组记录最短路,一个数组记录次短路. 每次对当前最短边,先更新最短路,更新不了最短路再更新次短路. 每条边处理两次,这样就可以在2n×n的复杂度内求得最短路和次短路了. #include<cstdio> #include<cstring> #include<queue> #include&l

最短路和次短路问题,dijkstra算法

1 /* 2  *题目大意: 3  *在一个有向图中,求从s到t两个点之间的最短路和比最短路长1的次短路的条数之和; 4  * 5  *算法思想: 6  *用A*求第K短路,目测会超时,直接在dijkstra算法上求次短路; 7  *将dist数组开成二维的,即dist[v][2],第二维分别用于记录最短路和次短路; 8  *再用一个cnt二维数组分别记录最短路和次短路的条数; 9  *每次更新路径的条数时,不能直接加1,,应该加上cnt[u][k],k为次短路径或者最短路径的标记; 10  *

poj3463 最短路和次短路

这道题就是让你求出有向图中最短路和比最短路长1的路的数量, 我们求出次短路和最短路的数量即可解决这道题 /* 求s到t的最短路与次短路(这里要求只比最短路多1)的条数之和 联想到最小,次小的一种更新关系: if(x<最小)更新最小,次小 else if(==最小)更新方法数 else if(x<次小)更新次小 else if(x==次小)更新方法数 同时记录s到u最短,次短路及方法数 用一个堆每次取最小的,更新完后再入堆 还是那个原理,第一次遇到的就是最优的,然后vi标记为真 方法数注意是加法

poj 3463 Sightseeing (dij 求最短路和次短路并计数)

dijkstra求最短路和次短路的求法和计算  模板 dijkstra求最短路的变形. 外循环要循环2*n-1次,因为dis[n][2]有2*n个状态,而dis[s][0]已经用过一次. 算法: 1.如果比最短路短就更新最短路和次短路. 2.如果和最短路相等,更新最短路的计数. 3.如果和次短路相等,更新次短路的方法数. 4.如果比次短路短,更新次短路. #include<cstdio> #include<iostream> #include<cstring> #inc

最短路与次短路计数

poj  3464  http://poj.org/problem?id=3463 问最短路的条数+比最短路权值大 1  的条数 做法  比较一下次短路和最短路的值  若次短路恰好比最短路大1,答案为最短路+次短路条数,否则答案就是最短路条数 1 #include<cstdio> 2 const int inf=0x3f3f3f3f; 3 class Count_short_path { ///最短路与次短路计数Dijkstra_o(MV^2) 4 typedef int typec;///边