ZOJ 3261 Connections in Galaxy War(逆向并查集)
参考链接:
http://www.cppblog.com/yuan1028/archive/2011/02/13/139990.html
http://blog.csdn.net/roney_win/article/details/9473225
题意:N个星球,编号从0到N-1。每个星球有一个战斗力power,且这N个星球之间建有一些通道,可以相互联系,在星球大战中,一些星球要向和自己联通的星球中power最强且大于自己的星球求救,且在星球大战中会有一些通道被损坏。
两种操作:破坏a和b之间的通道;a星球该向谁求救。
逆向并查集:
与并查集不同,给出一个图中原有的一些边,然后给出操作,操作不是向图中添加边,而是在已有的边上,将边删除。
对于该种情况,需要首先读入所有操作,把要求删除的边全部删除,再按照从后往前的顺序处理操作,这样删边操作转化为了添边的操作。
思路:我们先输入所有数据,即所谓的离线输入,将星球大战之后的状态作为初始状态,进行并查集的操作,遇到destroy时,就恢复a到b之间的通道,那么就很容易维护根节点的性质了,这题就变得很简单了。
#include <iostream>
#include <string.h>
#include <stdio.h>
#include <algorithm> using namespace std; const int maxn=;
int father[maxn];
long long power[maxn];
int n,m,q;
int ans[maxn*]; //存储最后要输出的答案 struct Edge{
int u,v,flag; //u、v表示边的端点,u<=v;flag=0表示改变被删除 //将u从小到大排序,若u相同时,按v从小到大排序
bool operator < (const Edge tmp)const {
if(u==tmp.u)
return v<tmp.v;
return u<tmp.u;
}
} edge[*maxn]; //先把所有的询问都存储下来
struct Query{
int k,a,b; //k=1表示操作是询问,k=2表示操作是删边
}query[maxn*]; struct Node {
int u; //u代表集合中具有最大能力值的点的编号
long p; //p代表集合中点的最大能力值
} node[maxn]; void init() {
for(int i=; i<n; i++) {
father[i]=i;
node[i].u=i;
node[i].p=power[i];
}
} int find_root(int x) {
if(father[x]!=x)
father[x]=find_root(father[x]);
return father[x];
} void Union(int x,int y) {
father[y]=x;
if(node[x].p<node[y].p){
node[x].p=node[y].p;
node[x].u=node[y].u;
}
else if(node[x].p==node[y].p && node[x].u>node[y].u){
node[x].u=node[y].u;
}
}
//二分查找边,a<=b
void findEdge(int a,int b){
int mid,mins=,maxs=m-;
while(mins<=maxs){
mid=(mins+maxs)>>;
if(a==edge[mid].u && b==edge[mid].v){
edge[mid].flag=;
return;
}
if(a<edge[mid].u){
maxs=mid-;
}
else if(a>edge[mid].u){
mins=mid+;
}
else{
if(b<edge[mid].v){
maxs=mid-;
}
else{
mins=mid+;
}
}
}
}
int main() {
int a,b,blank=;
char ch[];
while(scanf("%d",&n)!=EOF) {
memset(vis,,sizeof(vis));
if(blank==)
blank=;
else
puts("");
for(int i=;i<n;i++){
scanf("%I64d",&power[i]);
}
scanf("%d",&m);
for(int i=; i<m; i++) {
scanf("%d%d",&a,&b);
if(a<b) {
edge[i].u=a;
edge[i].v=b;
edge[i].flag=;
} else {
edge[i].u=b;
edge[i].v=a;
edge[i].flag=;
}
}
sort(edge,edge+m); //一开始忘记排序了
scanf("%d",&q);
for(int i=; i<q; i++) {
scanf("%s",ch);
if(ch[]=='q') {
scanf("%d",&a);
query[i].k=;
query[i].a=a;
query[i].b=-;
}
else{
scanf("%d%d",&a,&b);
int t;
if(a>b){
t=a;a=b;b=t;
}
findEdge(a,b);
query[i].k=;
query[i].a=a;
query[i].b=b;
}
}
int x,y;
init();
for(int i=;i<m;i++){
if(edge[i].flag==){
x=find_root(edge[i].u);
y=find_root(edge[i].v);
if(x!=y){
Union(x,y);
}
}
}
int u,v;
for(int i=q-;i>=;i--){
if(query[i].k==){
x=find_root(query[i].a);
if(node[x].p>power[query[i].a]){
ans[i]=node[x].u;
}
else{
ans[i]=-;
}
}
else{
u=query[i].a;
v=query[i].b;
x=find_root(u);
y=find_root(v);
if(x!=y){
Union(x,y);
}
ans[i]=-;
}
}
for(int i=;i<q;i++){
if(ans[i]==-)
continue;
printf("%d\n",ans[i]);
}
}
return ;
}
ZOJ 3261 Connections in Galaxy War(逆向并查集)的更多相关文章
- 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为切断 ...
- Connections in Galaxy War (逆向并查集)题解
Connections in Galaxy War In order to strengthen the defense ability, many stars in galaxy allied to ...
- ZOJ3261:Connections in Galaxy War(逆向并查集)
Connections in Galaxy War Time Limit: 3 Seconds Memory Limit: 32768 KB 题目链接:http://acm.zju.edu. ...
- Connections in Galaxy War(逆向并查集)
Connections in Galaxy War http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=3563 Time Limit ...
- 洛谷 P1197 BZOJ 1015 [JSOI2008]星球大战 (ZOJ 3261 Connections in Galaxy War)
这两道题长得差不多,都有分裂集合的操作,都是先将所有操作离线,然后从最后一步开始倒着模拟,这样一来,分裂就变成合并,也就是从打击以后最终的零散状态,一步步合并,回到最开始所有星球都被连为一个整体的状态 ...
- zoj 3261 Connections in Galaxy War
点击打开链接zoj 3261 思路: 带权并查集 分析: 1 题目说的是有n个星球0~n-1,每个星球都有一个战斗值.n个星球之间有一些联系,并且n个星球之间会有互相伤害 2 根本没有思路的题,看了网 ...
随机推荐
- 济南学习 Day 2 T3 pm
它[问题描述]N个人坐成一圈,其中第K个人拿着一个球.每次每个人会以一定的概率向左边的人和右边的人传球.当所有人都拿到过球之后,最后一个拿到球的人即为胜者.求第N个人获胜的概率. (所有人按照编号逆时 ...
- 关于CORS
前几天碰到CORS问题,只要在“Access-Control-Allow-Origin”响应头中添加对应域名即可. 今天做一个上传文件的demo,利用XMLHttpRequest向服务器发送post请 ...
- 状态模式(State)
状态模式,从字面意思上来讲应该是很简单的,就是针对实际业务上的内容,当类的内部的状态发生改变时,给出不同的响应体,就像现实中的人一样,早上没有吃饭,状态不好,上班.上课都会打哈欠,中午了,吃过午饭,又 ...
- HTML5读取本地文件 FileReader API接口
1.FileReader接口的方法 FileReader接口有4个方法,其中3个用来读取文件,另一个用来中断读取.无论读取成功或失败,方法并不会返回读取结果,这一结果存储在result属性中. Fil ...
- perl编程中的map函数示例
转自:http://www.jbxue.com/article/14854.html 发布:脚本学堂/Perl 编辑:JB01 2013-12-20 10:20:01 [大 中 小] 本文介绍 ...
- ASP.NET MVC4学习笔记之总体概述
断断续续使用ASP.NET MVC框架也有一年多了,也算积累了一些经验,唉,一直想写一些笔记好好总结一下,人太懒不想动笔,今天终于决定开始.希望自己能坚持下去. 这篇文章大体介绍ASP.NET MVC ...
- 用nodejs删除mongodb中ObjectId类型数据
mongodb中"_id"下面有个ObjectId类型的数据,想通过这个数据把整个对像删除,费了半天劲终于搞定费话少说上代码 module.exports = function ( ...
- SQLServer处理亿万级别的数据的优化措施
如何在SQLServer中处理亿万级别的数据(历史数据),可以按以下方面进行: 去掉表的所有索引 用SqlBulkCopy进行插入 分表或者分区,减少每个表的数据总量 在某个表完全写完之后再建立索引 ...
- Ubuntu中设置环境变量详解
1, 为单一用户:.bashrc: 为每一个运行bash shell的用户执行此文件.当bash shell被打开时,该文件被读取.打开用户主目录下的.bashrc,在这个文件中加入export PA ...
- [MySql] 设置了UTF8,中文存数据库中仍然出现问号
运行命令:SHOW VARIABLES LIKE 'character_set_%'; 结果 'character_set_client', 'utf8' 'character_set_connect ...