HDU 3749 Financial Crisis (点双连通+并查集)
<题目连接>
题目大意:
给你一个(保证输入无重边,无自环)无向图,然后有下面Q条询问,每条询问为:问你u点与v点之间有几条(除了首尾两点外,其他点不重复)的路径.如果有0条或1条输出0或1,如果有2条以上,输出”two or more”.
解题分析:
我们可以用并查集判断两点之间是否有路径相连通,如果两点不连通,则直接输出0即可。至于判断两点之间有几条不重复的路径相连,则是通过这两点是否属于同一点双连通分量来判断。不过需要注意的是,我们应该排除只有两个点的点双连通分量这一特殊情况。所以综上,只要待查询的两点属于点数大于2的点双连通分量,则直接输出"two or more"(在已经判断这两点连通的情况下),否则输出"one"。
#include <bits/stdc++.h>
using namespace std; #define pb push_back
#define rp(i, s, t) for (int i = s; i < t; i++)
#define clr(a, b) memset(a, b, sizeof(a))
const int N = 5e3 + ;
int n, m, q;
int tot, bcc;
int dfn[N], low[N], col[N], fa[N]; //col[i]表示的是这个点当前所属的点双连通分量的编号
vector<int> G[N], BCC[N], bl[N]; //bl[i]表示第i个节点属于的所有点双连通分量编号
struct Edge {
int u, v;
Edge(int u = , int v = ) : u(u), v(v) {}
};
void init() {
bcc = tot = ;
clr(dfn, );clr(col, ); clr(fa, -);
rp(i, , n) G[i].clear(), bl[i].clear();
}
stack<Edge> stk;
void Tarjan(int u, int fa) {
low[u] = dfn[u] = ++tot;
rp(i, , G[u].size()) {
int v = G[u][i];
if (v == fa) continue;
Edge e = Edge(u, v);
if (!dfn[v]) {
stk.push(e);
Tarjan(v, u);
low[u] = min(low[v], low[u]);
if (low[v] >= dfn[u]) { //割点
bcc++;
BCC[bcc].clear();
while (true) { //将栈中所有属于当前点双连通分量的点全部标记,注意,栈内存的是边
Edge x = stk.top(); stk.pop();
int st = x.u, ed = x.v;
if (col[st] != bcc) {
BCC[bcc].pb(st); //将u加入当前点双连通分量的节点集
col[st] = bcc; //标记u点当前所述的点双连通分量编号,防止在遍历同一点双连通分量的过程中重复将点标记
bl[st].pb(bcc); //一个点可能属于多个点双连通分量,bl[u]存储的是u点所属的多个点双连通分量
}
if (col[ed] != bcc) {
BCC[bcc].pb(ed);
col[ed] = bcc;
bl[ed].pb(bcc);
}
if (st == u && ed == v) break;
}
}
} else if (dfn[v] < dfn[u]) {
stk.push(e);
low[u] = min(low[u], dfn[v]);
}
}
}
int find(int i) {
if (fa[i] == -) return i;
return fa[i] = find(fa[i]);
}
int main() {
int ncase = ;
while (~scanf("%d%d%d", &n, &m, &q) && n) {
init();
while (m--) {
int u, v;
scanf("%d%d", &u, &v);
G[u].pb(v), G[v].pb(u); u = find(u), v = find(v); //并查集加边
if (u != v) fa[u] = v;
}
rp(i, , n) if (!dfn[i]) Tarjan(i, -);
printf("Case %d:\n", ++ncase);
while (q--) {
int u, v;
scanf("%d%d", &u, &v);
if (find(u) != find(v)) printf("zero\n"); //如果这两点不连通,直接输出0
else {
bool flag = false;
for (int i = ; i < bl[u].size() && !flag; i++) //遍历u和v所属的点双连通分量点集,判断是否有点双连通分量同时包含u和v
for (int j = ; j < bl[v].size() && !flag; j++) {
if (bl[u][i] == bl[v][j]) { //如果存在这样的点双连通分量
int ord = bl[u][i]; // ord代表该点双连通分量的编号
if (BCC[ord].size() > ) //要如果该点双连通分量中点数>2,才能说明u和v之间一定有2条即两条以上的道路
puts("two or more"), flag = true;
}
}
if (!flag) puts("one"); //如果不存在点数>2的点双连通分量同时包含u和v,则说明u和v之间只有一条边相连
}
}
}
}
2018-12-02
HDU 3749 Financial Crisis (点双连通+并查集)的更多相关文章
- HDU 3749 Financial Crisis(点-双连通分量)
Because of the financial crisis, a large number of enterprises go bankrupt. In addition to this, oth ...
- HDU 3749 Financial Crisis 经济危机(点双连通分量)
题意: 给一个图n个点m条边(不一定连通),接下来又q个询问,询问两个点是为“不相连”,“仅有一条路径可达”,“有两条及以上的不同路径可达”三种情况中的哪一种.注:两条以上的路径指的是路径上的点连1个 ...
- HDU 3749 Financial Crisis
Financial Crisis 题意:给一个图,包含N ( 3 <= N <= 5000 )个点, M ( 0 <= M <= 10000 )条边 and Q ( 1 < ...
- HDU 1598 find the most comfortable road 并查集+贪心
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1598 find the most comfortable road Time Limit: 1000 ...
- HDU 1116 Play on Words(欧拉回路+并查集)
传送门: http://acm.hdu.edu.cn/showproblem.php?pid=1116 Play on Words Time Limit: 10000/5000 MS (Java/Ot ...
- BZOJ4998 星球联盟(LCT+双连通分量+并查集)
即要求动态维护边双.出现环时将路径上的点合并即可.LCT维护.具体地,加边成环时makeroot+access+splay一套把这段路径提出来,暴力dfs修改并查集祖先,并将这部分与根断开,视为删除这 ...
- ACM: hdu 1811 Rank of Tetris - 拓扑排序-并查集-离线
hdu 1811 Rank of Tetris Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & % ...
- HDU(1856),裸的带权并查集
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1856 题意:朋友圈问题,A和B是朋友,B和C是朋友则A和C也是朋友,依次类推,题目的意思就是求最大的朋 ...
- HDU 1213 How Many Tables(模板——并查集)
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1213 Problem Description Today is Ignatius' birthday ...
随机推荐
- 软件测试-历史bug回顾(持续更新~)
1.第一次编写程序的时候非常不容易看出的错误就是 = 和 == 的区别. 赋值符号,判断符号之间有着天差地别.导致一直循环错误,一直苦苦找寻愿意无果,一步步看代码解决. 2.对于终结条件判断不是很清楚 ...
- oracle 11g完全彻底的卸载
1.关闭oracle所有的服务.可以在windows的服务管理器中关闭: 2.打开注册表:regedit 打开路径: <找注册表 :开始->运行->regedit> HKEY ...
- Confluence 6 上传站点图标后重置你的配色方案
当你上传一个站点标识图片后,Confluence 会根据你上传的图片文件自动侦测使用的颜色,并为你设置自动配色方案. 你可以按照上面描述的方法修改色彩配色方案,或者你也可以重置配色方案为默认的配色方案 ...
- javaScript遍历对象、数组总结
javaScript遍历对象总结 1.使用Object.keys()遍历 返回一个数组,包括对象自身的(不含继承的)所有可枚举属性(不含Symbol属性). var obj = {'0':'a ...
- day10 函数
函数基础 函数概念 为什么用函数 什么是函数 定义函数 调用函数 返回值详解上节回顾 文件处理 修改文件 1.读取要修改文件内容 read可能导致内存溢出 赋值给变量 修改 将修改后的内容重新写入文件 ...
- linux符号与正则表达式
特殊符号 > 或者1> 标准输出重定向 先把文件的内容清空 再放入新的内容 >> 或 2>> 追加重定向 把内容放入文件的最后一行 1 ...
- Git使用三:git的使用流程
先创建仓库 创建一个文件夹,作为仓库使用 初始化仓库,在仓库目录的命令行下输入:git init 第一步:创建一个文件,并写入内容,这里写入内容的时候,要把编码设置为utf-8 第二步:输入命令将文件 ...
- 基于“MVC”框架集设计模式,开发用户管理系统!
MVC----(Model View Controller)设计模型: M:表示业务数据和业务规则.包括DAO(beans).DBHelper(DBHelper),用于封装数据库连接,业务数据库处理. ...
- markdown 相关零碎知识
1.尖括号<>在markdown会被当做html符号,解决办法:用转义字符,如:<测试> 可以写作<:测试>
- C#学习-属性是对字段的扩展
属性是对字段的扩展. 根据面向对象语言的封装思想,字段最好设为private,因为这样可以防止客户端直接对字段进行篡改,从而保证了内部成员的完整性. 于是为了访问类中的私有字段,C#提供了属性这种机制 ...