ZOJ3261:Connections in Galaxy War(逆向并查集)
Connections in Galaxy War
Time Limit: 3 Seconds Memory Limit: 32768 KB
题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3261
Description:
In order to strengthen the defense ability, many stars in galaxy allied together and built many bidirectional tunnels to exchange messages. However, when the Galaxy War began, some tunnels were destroyed by the monsters from another dimension. Then many problems were raised when some of the stars wanted to seek help from the others.
In the galaxy, the stars are numbered from 0 to N-1 and their power was marked by a non-negative integer pi. When the star A wanted to seek help, it would send the message to the star with the largest power which was connected with star A directly or indirectly. In addition, this star should be more powerful than the star A. If there were more than one star which had the same largest power, then the one with the smallest serial number was chosen. And therefore, sometimes star A couldn't find such star for help.
Given the information of the war and the queries about some particular stars, for each query, please find out whether this star could seek another star for help and which star should be chosen.
Input:
There are no more than 20 cases. Process to the end of file.
For each cases, the first line contains an integer N (1 <= N <= 10000), which is the number of stars. The second line contains N integers p0, p1, ... , pn-1 (0 <= pi <= 1000000000), representing the power of the i-th star. Then the third line is a single integer M (0 <= M <= 20000), that is the number of tunnels built before the war. Then M lines follows. Each line has two integers a, b (0 <= a, b <= N - 1, a != b), which means star a and star b has a connection tunnel. It's guaranteed that each connection will only be described once.
In the (M + 2)-th line is an integer Q (0 <= Q <= 50000) which is the number of the information and queries. In the following Q lines, each line will be written in one of next two formats.
"destroy a b" - the connection between star a and star b was destroyed by the monsters. It's guaranteed that the connection between star a and star b was available before the monsters' attack.
"query a" - star a wanted to know which star it should turn to for help
There is a blank line between consecutive cases.
Output
For each query in the input, if there is no star that star a can turn to for help, then output "-1"; otherwise, output the serial number of the chosen star.
Print a blank line between consecutive cases.
Sample Input
2
10 20
1
0 1
5
query 0
query 1
destroy 0 1
query 0
query 1
Sample Output
1
-1
-1
-1
题意:
先进行合并操作,然后进行询问和分裂操作,询问操作返回相连值最大的那个点的下标,如果值最大的点有多个,则要下标最小;分裂操作就从a,b中间断开。
题解:
因为并查集只能“并”,不能“裂”,所以直接处理不好处理。
但是如果我们反过来看,每次分裂操作都是一次合并操作,刚好可以用并查集处理。
之后,我们维护下集合中最大的值以及最小坐标就行了。
代码如下:
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <iostream>
#include <map>
using namespace std; typedef long long ll;
const int N = ,Q = ;
int n,m,que;
char s[Q][];
ll P[N];
struct father{
int fa,id;
ll p;
}f[N];
struct query{
int x,y;
}q[Q];
struct link{
int p1,p2;
}l[N<<];
//用很多数组来储存...
int find(int x){
return f[x].fa==x ? x : f[x].fa=find(f[x].fa);
}
void Union(int x,int y){
int fx=find(x),fy=find(y);
f[fx].fa=fy;
if(f[fx].p>f[fy].p){
f[fy].p=f[fx].p;
f[fy].id=f[fx].id;
}else if(f[fx].p==f[fy].p){
f[fy].id=min(f[fy].id,f[fx].id);
}
}
int main(){
bool flag = false;
while(~scanf("%d",&n)){
map <int ,map<int,int> > mp;
for(int i=;i<=n;i++) f[i].fa=i,f[i].id=i;
for(int i=;i<n;i++) scanf("%lld",&f[i].p),P[i]=f[i].p;
scanf("%d",&m);
for(int i=,tmp1,tmp2;i<=m;i++){
scanf("%d%d",&tmp1,&tmp2);
if(tmp1>tmp2) swap(tmp1,tmp2);
l[i].p1=tmp1;l[i].p2=tmp2;
}
scanf("%d",&que);
for(int i=,tmp1,tmp2;i<=que;i++){
scanf("%s",s[i]);
if(s[i][]=='d'){
scanf("%d%d",&tmp1,&tmp2);
if(tmp1>tmp2) swap(tmp1,tmp2);
q[i].x=tmp1;q[i].y=tmp2;
mp[tmp1][tmp2]=;
}else scanf("%d",&q[i].x);
}
for(int i=;i<=m;i++){
int p1=l[i].p1,p2=l[i].p2;
if(mp[p1][p2]) continue;
int fx=find(p1),fy=find(p2);
if(fx==fy) continue;
Union(p1,p2);
}
int ans[Q];int tot=;
for(int i=que;i>=;i--){
if(s[i][]=='d') Union(q[i].x,q[i].y);
else{
int now = q[i].x;
int fx=find(now);
if(f[fx].p<=P[now] ) ans[++tot]=-;
else ans[++tot]=f[fx].id;
}
}
if(flag) printf("\n");
else flag=true ;
for(int i=tot;i>=;i--) printf("%d\n",ans[i]);
}
return ;
}
ZOJ3261:Connections in Galaxy War(逆向并查集)的更多相关文章
- Connections in Galaxy War (逆向并查集)题解
Connections in Galaxy War In order to strengthen the defense ability, many stars in galaxy allied to ...
- Connections in Galaxy War(逆向并查集)
Connections in Galaxy War http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=3563 Time Limit ...
- ZOJ 3261 Connections in Galaxy War(逆向并查集)
参考链接: http://www.cppblog.com/yuan1028/archive/2011/02/13/139990.html http://blog.csdn.net/roney_win/ ...
- ZOJ3261 Connections in Galaxy War —— 反向并查集
题目链接:https://vjudge.net/problem/ZOJ-3261 In order to strengthen the defense ability, many stars in g ...
- zoj 3261 Connections in Galaxy War(并查集逆向加边)
题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3261 题意:有很多颗星球,各自有武力值,星球间有一些联系通道,现 ...
- ZOJ 3261 - Connections in Galaxy War ,并查集删边
In order to strengthen the defense ability, many stars in galaxy allied together and built many bidi ...
- ZOJ - 3261 Connections in Galaxy War(并查集删边)
https://cn.vjudge.net/problem/ZOJ-3261 题意 银河系各大星球之间有不同的能量值, 并且他们之间互相有通道连接起来,可以用来传递信息,这样一旦有星球被怪兽攻击,便可 ...
- ZOJ 3261 Connections in Galaxy War (逆向+带权并查集)
题意:有N个星球,每个星球有自己的武力值.星球之间有M条无向边,连通的两个点可以相互呼叫支援,前提是对方的武力值要大于自己.当武力值最大的伙伴有多个时,选择编号最小的.有Q次操作,destroy为切断 ...
- ZOJ-3261 Connections in Galaxy War 并查集 离线操作
题目链接:https://cn.vjudge.net/problem/ZOJ-3261 题意 有n个星星,之间有m条边 现一边询问与x星连通的最大星的编号,一边拆开一些边 思路 一开始是真不会,甚至想 ...
随机推荐
- python递归函数(计算阶乘)
def f1(x,x1=1): if x == 1: return x1 #x1这个值为我们所需要的值,所以返回 x1 *= x r = f1(x-1,x1) #r接收返回值,并在下面接着返回 ret ...
- ruby 数据类型Number
Ruby支持的数据类型包括基本的Number.String.Ranges.Symbols,以及true.false和nil这几个特殊值,同时还有两种重要的数据结构——Array和Hash 数值类型(N ...
- Jetson tx1 安装cuda错误
前两天安装Jetpack3.0的时候,看着网上的教程以为cuda会自动安装上,但是历经好几次安装,都安装不上cuda,也刷了好几次jetpack包.搜遍了网上的教程,也没有安装上.错误如下图所示: 这 ...
- UVA 1593 Alignment of Code(紫书习题5-1 字符串流)
You are working in a team that writes Incredibly Customizable Programming Codewriter (ICPC) which is ...
- Ubuntu 首次给root用户设置密码
用过ubuntu的人都知道,刚安装好root用户是没有密码的,没有密码我们就没法用root用户登录.给root用户设置密码输入命令sudo passwd root,然后系统会让你输入密码,这时输入的密 ...
- 【LoadRunner】解决LR11无法录制Chrome浏览器脚本问题
LoadRunner录制脚本时,遇到高版本的IE.FireFox,或者Chrome浏览器,会出现无法录制脚本的问题,下面就来讲一下如何利用LR自带的wplus_init_wsock.exe插件进行脚本 ...
- linux ----- Vim进入和退出命令
Vim进入和退出命令 本来不想写任何关于vim的文章的,无奈我今天又忘记怎么退出vim了,常用命令是ESC,然后:wq(保存并退出),:q!(不保存并强制退出),i进入vim模式.另外还有其它 ...
- 51单片机数码管字符H自右向左移动
#include <reg51.h> #define uint unsigned int #define uchar unsigned char sfr P0M0 = 0x94; sfr ...
- libvirt保持虚拟机运行情况下修改名称
通过virsh list命令能看到虚拟机的列表: [root@compute2 ~]# virsh list Id 名称 状态 ------------------------------------ ...
- 使用CodeBlocks为你的程序添加程序文件图标和启动读入图标
其实也非常简单,自己这两天用win32api做了一个小程序,可是发现图标却是dos的,太难看了,于是就想起以前学win32汇编时候用到的工具,ResEd,已经被我汉化了一些,估计有新的版本发布吧,但是 ...