最短路计数- C++

最短路计数

题目描述 给出一个 N 个顶点 M 条边的无向无权图,顶点编号为1−N。问从顶点1开始,到其他每个点的最短路有几条。
输入格式 第一行包含2个正整数N,M,为图的顶点数与边数。
接下来M行,每行2个正整数x,y,表示有一条顶点 x 连向顶点 y 的边,请注意可能有自环与重边。
输出格式 共N行,每行一个非负整数,第 i 行输出从顶点1到顶点 i 有多少条不同的最短路,由于答案有可能会很大,你只需要输出 ans  mod  100003 后的结果即可。如果无法到达顶点 i 则输出 0 。
输入输出样例 输入 #1
5 7
1 2
1 3
2 4
3 4
2 3
4 5
4 5
输出 #1
1
1
1
2
4

这题就是得注意起点是1,而如果在未遍历该节点时,此时该节点的路径就是等于它前一个节点路径。若已经遍历过该节点,则此时节点路径便要加上上一个节点的路径。
C++代码:

#include<iostream>
#include<cstdio>
#include<queue>
#define maxPath 999999999
using namespace std;
int dis[1000001],cnt;
int book[1000001];
int ans[1000001];
int head[1000001];
queue <int> q;
struct edge{
   int v;
   int next;
};
edge e[2000001];
void add(int u, int v)
{
   cnt ++;
   e[cnt].v = v;
   e[cnt].next = head[u];
   head[u] = cnt; 
}
int main()
{
   int N,M;
   cin >> N >> M;
   for(int i = 1; i <= N; i ++)
   dis[i] = maxPath;
   dis[1] = 0;
   for(int i = 1; i <= M; i ++)
   {
    int u,v;
    scanf("%d%d",&u,&v);
      add(u,v);
      add(v,u);
   }
q.push(1);
   book[1] = 1;
   ans[1] = 1;
   int k = 0;
   while(q.size())
   { 
    k = q.front();
    q.pop();
    for(int i = head[k]; i != 0; i = e[i].next)
    {
        if(dis[e[i].v] > dis[k] + 1)
        {
           dis[e[i].v] = dis[k] + 1;
   ans[e[i].v] = ans[k];
   
   //判断改点是否入了队列
   if(book[e[i].v] == 0)
   {
     q.push(e[i].v);
     book[e[i].v] = 1;  
   } 
     }else
     if(dis[e[i].v] == dis[k] + 1)
     {
       ans[e[i].v] += ans[k];
       ans[e[i].v] %= 100003;
     }
 }
   }
   for(int i = 1; i <= N; i ++)
   cout << ans[i] << endl;
   return 0; 
} 
发布了26 篇原创文章 · 获赞 0 · 访问量 1349

猜你喜欢

转载自blog.csdn.net/weixin_43846217/article/details/104149344