题目链接

描述

乱世天下,诸侯割据。每个诸侯王都有一片自己的领土。但是不是所有的诸侯王都是安分守己的,实力强大的诸侯国会设法吞并那些实力弱的,让自己的领土面积不断扩大。而实力弱的诸侯王为了不让自己的领土被吞并,他会联合一些其他同样弱小的诸侯国,组成联盟(联盟不止一个),来共同抵抗那些强大的诸侯国。 强大的诸侯国为了瓦解这些联盟,派出了最优秀的间谍来离间他们,使一些诸侯国退出联盟。最开始,每个诸侯国是一个联盟。

有两种操作

1、U x y 表示x和y在同一个联盟。(0≤x,y<n)

2、D x   表示x退出联盟。

  • 输入

    多组测试数据第一行两个数,n和m(1 ≤ n≤ 10^5, 1 ≤ m ≤10^5),分别表示诸侯国的个数和操作次数。接下来有m行操作
  • 输出

    输出联盟的个数
  • 样例输入

    5 7

    U 0 1

    U 1 2

    U 0 3

    D 0

    U 1 4

    D 2

    U 0 2

    10 1

    U 0 9
  • 样例输出

    Case #1: 2

    Case #2: 9

分析:

首先知道是用并查集写的没有商量,但是写着写着发现自己的思路通不过去,然后就百度了大神的题解,先看一下大神的思路吧。

贴一下大神思路(地址):

这道题一读题,应该都能想到要用并查集归并集合。这道需要实现并查集的删除操作。那么问题就来了,并查集的的结构是一颗树,它的边是有向且只指向父节点的。那么删除一个节点(也就是让它的父节点成为它自己),指向这个节点孩子节点的根就会丢失。学习这个算法的时候网上说是用虚根,看了很久才看懂。

我就想用通俗更易懂的描述出来”虚根“:

例子:食品店要给顾客甲派送食物food装在箱子box里,box有个挂钩(挂钩就相当于连接父节点的边)。food[]存储箱子编号,box[]存父节点。

food有很多,把要送的归在一类后。顾客甲打电话退订了某些。

如下图,food[2]=2.编为2的food它的箱子box编号是2

box[3]=3;编号为3的箱子box它的挂钩挂在自己上(它的父节点是它自己)。

建立如下并查集树。box[3]=2;

然后顾客甲打电话要退订编号为4,6的food。

接下来我们只需要把编号为4的food拿走,用编号为n++(7)的箱子装起来。

food[4]=7;

box[7]=[7];

编号为4的箱子依然留在那里,这样就不影响编号4的box后面挂的箱子的根节点就不会丧失。

拿走6同理。

food[6]=8;

box[8]=8;

接下来又有一顾客乙要走了4,6.

box[food[6]]=food[4];\把编号为6的food它所在的箱子8的挂钩挂到编号为4的food它所在的箱子7上。

food 2,3,1,5归顾客甲一类,food 4,6归顾客乙一类。这样虽然浪费了盒子但是归类是正确的。搜索x代表元,也就是通过x的箱子找到根箱子。

比一般的并查集多了一个删除节点的操作,肯定不能够在原来的父节点上操作,所以额外多开一个数组来处理。

代码:

#include<stdio.h>
#include<string.h>
#include<iostream>
#include<algorithm>
#include<stack>
#include<math.h>
using namespace std;
int parent[200009];
int b[200009];
int vis[200009];
int n,m;
int ans;
int add;
int init()///初始化
{
for(int i=0; i<=n; i++)
{
parent[i]=i;
b[i]=i;
}
add=n;
ans=0;
memset(vis,0,sizeof(vis));
} int Find(int a)
{
if(parent[a]==a)
return a;
else
return parent[a]=Find(parent[a]);
} void He(int a,int b)///合并节点
{
int x=Find(a);
int y=Find(b);
if(x!=y)
{
parent[x]=y;
}
} void Remove(int x)///删除节点x, {
b[x]=add;
parent[add]=add;
add++;
}
int main()
{
int kase=1;
while(~scanf("%d%d",&n,&m))
{
init();
char ch;
int a,bb;
while(m--)
{
scanf(" %c",&ch);
if(ch=='U')
{
scanf(" %d%d",&a,&bb);
He(b[a],b[bb]);///将节点a和bb合并,但是合并的时候并不是合并a和bb,而是将他们的父节点合并
}
if(ch=='D')
{
scanf("%d",&a);
Remove(a);
}
}
int vis[200001]= {0},ans=0;
for(int i=0; i<n; i++)
{
int mmm=Find(b[i]);
if(vis[mmm]==0)
{
ans++,vis[mmm]=1;
}
}
printf("Case #%d: %d\n",kase++,ans);
}
return 0;
}

NYOJ 1022 合纵连横 (并查集)的更多相关文章

  1. nyoj 1022 合纵连横 (并查集<节点删除>)

    合纵连横 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 乱世天下,诸侯割据.每个诸侯王都有一片自己的领土.但是不是所有的诸侯王都是安分守己的,实力强大的诸侯国会设法 ...

  2. nyoj 1022 合纵连横 经典并查集

    思路:关键在于并查集的删点操作. 给每个诸侯国一个另外的编号,比如box[i]表示诸侯国i现在处于第box[i]个联盟,可以随时改变它的联盟编号,并且让box[i] = k, 实现删除操作.以前联盟中 ...

  3. Nyoj 布线问题(并查集&&图论)

    描述南阳理工学院要进行用电线路改造,现在校长要求设计师设计出一种布线方式,该布线方式需要满足以下条件:1.把所有的楼都供上电.2.所用电线花费最少   输入 第一行是一个整数n表示有n组测试数据.(n ...

  4. nyoj 711 枚举+并查集

     #include<stdio.h>//从大到小不断枚举边直到找到s-t的路径,判断从s可以到t可以用并查集来判断 #include<stdlib.h>//枚举最大的一条边肯定 ...

  5. nyoj 1022 合纵连横【并查集节点的删除】

    合纵连横 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 乱世天下,诸侯割据.每个诸侯王都有一片自己的领土.但是不是所有的诸侯王都是安分守己的,实力强大的诸侯国会设法 ...

  6. nyoj 1022:合纵连横(并查集删点)

    题目链接 参考链接 只附代码好了 #include<bits/stdc++.h> using namespace std; ; int a[N],b[N],vis[N]; int n,m, ...

  7. nyoj 38 简单并查集的应用&最小生成树

    #include<stdio.h> #include<stdlib.h> #define inf 0x3fffffff #define N 600 struct node { ...

  8. NYOJ 208 Supermarket (模拟+并查集)

    题目链接 描述 A supermarket has a set Prod of products on sale. It earns a profit px for each product x∈Pr ...

  9. NYOJ 129 树的判定 (并查集)

    题目链接 描述 A tree is a well-known data structure that is either empty (null, void, nothing) or is a set ...

随机推荐

  1. bond下改变网卡

    浪潮服务器打开控制台 用ip addr查看哪个网卡是绑定的,eth2和eth4是绑定状态 用mv命令,更改网卡名称 并将每个网卡里的信息更改 reboot,重启 ip addr查看,eth6和eth8 ...

  2. ctf实验平台-成绩单

    题目链接:http://120.24.86.145:8002/chengjidan/ 平台地址:http://123.206.31.85/ 第一步:暴库 id=-1' union select 1,2 ...

  3. 免费各种查询API接口

    快递查询 http://www.kuaidi100.com/query?type=quanfengkuaidi&postid=390011492112 (PS:快递公司编码:申通"s ...

  4. sleep() 与 wait()的比较

    1.这两个方法来自不同的类分别是,sleep来自Thread类,和wait来自Object类. sleep是Thread的静态类方法,谁调用的谁去睡觉,即使在a线程里调用了b的sleep方法,实际上还 ...

  5. Maven学习——1、安装与修改Maven的本地仓库路径

    1.1.下载 官网 http://maven.apache.org/download.cgi 1.2.安装配置 apache-maven-3.3.3-bin.zip 解压下载的压缩包 1.3.配置环境 ...

  6. 第97天:CSS3渐变和过渡详解

    一.渐变 渐变是CSS3当中比较丰富多彩的一个特性,通过渐变我们可以实现许多炫丽的效果,有效的减少图片的使用数量,并且具有很强的适应性和可扩展性. 可分为线性渐变.径向渐变 1. 线性渐变 (grad ...

  7. 【bzoj4487】[Jsoi2015]染色问题 容斥原理

    题目描述 棋盘是一个n×m的矩形,分成n行m列共n*m个小方格.现在萌萌和南南有C种不同颜色的颜料,他们希望把棋盘用这些颜料染色,并满足以下规定: 1.  棋盘的每一个小方格既可以染色(染成C种颜色中 ...

  8. 秒杀多线程第八篇 经典线程同步 信号量Semaphore (续)

    java semaphore实现: Semaphore当前在多线程环境下被扩放使用,操作系统的信号量是个很重要的概念,在进程控制方面都有应用.Java 并发库 的Semaphore 可以很轻松完成信号 ...

  9. Tribles UVA - 11021(全概率推论)

    题意: 有k只麻球,每只只活一天,临死之前可能会出生一些新的麻球, 具体出生i个麻球的概率为P,给定m,求m天后麻球全部死亡的概率. 解析: 从小到大,先考虑一只麻球的情况  设一只麻球m天后全部死亡 ...

  10. Day20-单表中获取表单数据的3种方式

    1. 搭建环境请参考:http://www.cnblogs.com/momo8238/p/7508677.html 2. 创建表结构 models.py from django.db import m ...