HDU 5876 Sparse Graph(补图上BFS)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5876

题意:

  有一个 n 个点无向图,再给你 m 对顶点, 代表着这 m 对顶点之间没有边, 除此之外每两个点之间都有一条边, 且权值为 1.然后还有一个源点 S, 让你计算源点到其他各点之间的最短距离,如果不存在则输出 -1.也就是说让你在所给的图的补图上求源点到其他各点的最短路径.

思路:

  补图上求最短路径算是比较经典的题.在这里所求的最短路其实并不需要用到 dijkstra 之类的算法,由于每条边之间的距离都为 1,每条边的权值一样.那么就可以想到这个做法:

  步骤1:根据题意建图.然后建立一个队列,把源点 S 压入队列,其他的各个点存到另一个集合 V 里, 并建立一个数组来存储源点到其他各点的最短距离,初始化为 -1, dis[S] = 0.

  步骤2:从队列首部取出一个节点 v,访问所有与节点 v 不相邻的节点 u,把 u 压入到队列尾部, 并从集合 V 中把 u 除去, 更新dis[u] = dis[v] + 1.

  步骤3:反复重复步骤 2, 直到队列为空.之后 dis 数组里保存的就是答案.

  解决了怎么做之后,还没有完,由于题目给的点数非常大,所以还需要优化下时间.这里比较费时间的就是步骤 2 中的找不相邻的点.这里就可以用 STL 中的 set 来维护集合 V, 也就是未被访问到的点. 初始化 set1 中为所有点(除去源点 S), 当访问的点 v 时, 在 set1 中除去与点 v 相邻的点 u, 并加入到 set2 中,那么 set1 中剩下的点就是所有与点 v 不相邻的点, 依次遍历压入队列. 之后再把 set2 拷贝到 set1, 那么 set1 就是剩下的未被访问到的点,如此反复下去.可以看到,对于每条边只访问一次.每个点也只进一次队列.所以总的时间复杂度为 O(n * m),可以达到要求了.

notes:建立 无向图 的时候每条边要存两次. 所以数组的大小一定是原题所给的边数的两倍! 两倍! 两倍!顺便求一个用链表实现的代码,自己用链表没写出来.(我好菜啊.jpg

代码:

 1 #include <iostream>
 2 #include <cstdio>
 3 #include <queue>
 4 #include <set>
 5 #include <cstring>
 6 #include <algorithm>
 7 
 8 using namespace std;
 9 typedef long long LL;
10 
11 const int MAXN = 200000;
12 const int MAXE = 20000;
13 int n, m, T, S;
14 int dis[MAXN + 3];//保存最终的最短距离
15 
16 int head[MAXN + 3], len; //链式前向星
17 struct NODE {int to; int next; };
18 NODE edge[2 * MAXE + 3];
19 
20 void addedge(int u, int v) { //链式前向星加边
21     edge[len].to = v;
22     edge[len].next = head[u];
23     head[u] = len++;
24 }
25 
26 void BFS() { //从起点开始BFS
27     memset(dis, -1, sizeof(dis));
28     queue <int> Qu;
29     Qu.push(S); dis[S] = 0; //起点初始化
30     set<int> unsed, hep;    //用来维护尚未被访问的点
31     for(int i = 1; i <= n; i++) unsed.insert(i);
32     unsed.erase(S);
33     while( !Qu.empty() ) {
34         int tp = Qu.front(); Qu.pop();
35         for(int k = head[tp]; k != -1; k = edge[k].next) { //从 unsed 中除去和当前拓展节点相邻的点,同时加入到临时辅助的集合中
36             if(unsed.find(edge[k].to) != unsed.end()) {
37                 unsed.erase(edge[k].to);
38                 hep.insert(edge[k].to);
39             }
40         }
41         for(set<int>::iterator it = unsed.begin(); it != unsed.end(); it++) {// unsed 暂时保存的是和当前拓展节点不相邻的点
42             Qu.push(*it);
43             dis[*it] = dis[tp] + 1;
44         }
45         hep.swap(unsed), hep.clear();//从辅助集合中 copy 剩下的未被访问到的点.
46     }
47 }
48 
49 int main() {
50     scanf("%d", &T);
51     while(T--) {
52         memset(head, -1, sizeof(head));
53         scanf("%d%d", &n, &m);
54         int u, v; len = 0;
55         for(int i = 0, len = 0; i < m; i++) {
56             scanf("%d%d", &u, &v);
57             addedge(u, v); addedge(v, u);
58         }
59         scanf("%d", &S);
60         BFS();
61         for(int i = 1, j = 0; i <= n; i++) if(i != S) printf("%d%c", dis[i], " \n"[++j == n - 1]);
62     }
63     return 0;
64 }

 

转载于:https://www.cnblogs.com/Ash-ly/p/5876199.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值