zoukankan      html  css  js  c++  java
  • 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], " 
    "[++j == n - 1]);
    62     }
    63     return 0;
    64 }
  • 相关阅读:
    myEclipse Debug
    C# DataTable的詳細使用方法
    算法 《秦九韶算法java实践》
    【闲聊产品】之五:谁来背黑锅?
    ubuntu install mysql server method
    H264解码的一个測试程序
    Struts2自己定义拦截器实例—登陆权限验证
    【剑指offer】二叉树的镜像
    ubuntu12.04下搭建ftpserver
    C++Vector使用方法
  • 原文地址:https://www.cnblogs.com/Ash-ly/p/5876199.html
Copyright © 2011-2022 走看看