题意:
有N封邮件, 然后又两种操作,
如果是M X Y , 表示X和Y是相同的邮件。
如果是S X,那么表示对X的判断是错误的,X是不属于X当前所在的那个集合,要把X分离出来,让X变成单独的一个。
最后问集合的个数。

方法一:设立虚父节点

思路:n~n+n-1作为一开始初始化的根节点,而0~n-1作为虚拟根节点(即初试时它们指向n~n+n-1),之后删除节点操作时用n+n-1~n+n+m作为备用节点。

   删除时直接修改0~n-1指向的节点(即0~n-1的父亲的值)变为备用节点

  设cnt为备用节点,假如一个集合中1是这个集合的父节点(其实是虚的,因为它还指向一开始初始化的n+1,这才是该集合的真正的根节点)。

  子节点有2,3,它们分别先指向n+2,n+3,然后都指向实际的根节点即n+1。

1.假如我删除的是1,则我只要令father[1]=cnt,这样2、3指向的还是n+1,它们处于同一个集合内,而1指向的是cnt,已经不处于和2、3同样的集合了。然后cnt++;

   2.假如我删除的是2,则我令father[2]=cnt,同样1、3指向的还是n+1,而2的父亲变为cnt,把它们分离开来了。然后cnt++。

求集合个数有两种:用set来求最后集合的个数,640ms;用vis来求最后集合的个数,562ms

#include <iostream>
#include <stdio.h>
#include <string.h>
#include <set> using namespace std;
set<int> group;
int father[];
int vis[];
int n,m,cnt,ans; void init(){
memset(vis,,sizeof(vis));
cnt=*n;
ans=;
for(int i=;i<n;i++)
father[i]=n+i;
for(int i=n;i<=*n+m;i++)
father[i]=i;
group.clear();
} int find_root(int x){
if(father[x]!=x)
father[x]=find_root(father[x]);
return father[x];
}
void Union(int a,int b){
int x=find_root(a);
int y=find_root(b);
father[y]=x;
} int main()
{
char ch[];
int a,b,t=;
while(scanf("%d%d",&n,&m)!=EOF){
if(n== && m==){
break;
}
init();
t++;
for(int i=;i<=m;i++){
scanf("%s",ch);
if(ch[]=='M'){
scanf("%d%d",&a,&b);
Union(a,b);
}
else{
scanf("%d",&a);
father[a]=cnt++; //直接改变它的父亲为备用节点即可。
}
}
/*
for(int i=0;i<n;i++){
int fa=find_root(i);
group.insert(fa);
}
*/
for(int i=;i<n;i++){
int fa=find_root(i);
if(!vis[fa]){
ans++;
vis[fa]=; //fa为一个根节点,之后如果还有点的根节点为fa,表明在同一个集合中,不用再重复加了。
}
} //printf("Case #%d: %d\n",t,group.size());
printf("Case #%d: %d\n",t,ans);
} return ;
}

方法二:穿个马甲/找个代理:

思路:当要删除x点时,不是真的删除x点, 而是通过映射方式(这里用数组majia[N]),把x变成一个新的点即majia[x]=cnt.

  看到网上有这么解释的:用了代理majia[i]表示i的代理是majia[i],然后以后操作的时候每个人干活只找代理干,即查找、合并传递的参数都是majia[i],而不是i。

  然后删除的时候直接换个新代理(这个代理不能和别人的代理冲突),majia[i]=cnt++;

  设cnt备用点,假如一个集合中1是这个集合的父节点,子节点有2,3。合并的时候传递的其实是majia[1],majia[2],majia[3], 但是因为他们一开始即是本身,所以和传统的并查集没什么不同。

  目前father[1]=1,father[2]=1,father[3]=1,majia[1/2/3]=majia[1/2/3]

1.假如我删除的是1,则我只要令majia[1]=cnt++;这样如果以后我想要查找1的根节点时,我实际上传递的值为cnt,而father[cnt]=cnt,因此相当于1的根节点变为n了。

  而查找2、3的根节点时,传递的还是2、3,所以实际上等同于1和2、3分离了。

  然后假如1和4合并,那么传递的其实值为(majia[1],majia[4]),即(cnt,4),那么father[cnt]=4,即1、4的根节点为4.

2.假如我删除的是2,则我令majia[2]=cnt,此时,我若查找2的根节点,传递的参数为cnt,则结果为cnt,不为1。也就相当于2从1所在的集合删去了。

  2还是2,只不过相当于穿了一层外套,换了另一种身份,我们用这种身份来表示2。

#include <iostream>
#include <stdio.h>
#include <string.h>
#include <set> using namespace std; set<int> group; //可以用set来获取集合个数(因为set里元素不可重复)
int father[];
int majia[]; //代理,或者理解为第二个身份
int vis[]; //最后用于获取集合个数
int n,m,cnt,ans; void init(){
memset(vis,,sizeof(vis));
cnt=n;
ans=;
for(int i=;i<=n+m;i++)
father[i]=majia[i]=i;
group.clear();
} int find_root(int x){
if(father[x]!=x)
father[x]=find_root(father[x]);
return father[x];
}
void Union(int a,int b){
int x=find_root(a);
int y=find_root(b);
father[y]=x;
} int main()
{
char ch[];
int a,b,t=;
while(scanf("%d%d",&n,&m)!=EOF){
if(n== && m==){
break;
}
init();
t++;
for(int i=;i<=m;i++){
scanf("%s",ch);
if(ch[]=='M'){
scanf("%d%d",&a,&b);
Union(majia[a],majia[b]); //操作时,即传递参数,都是用它的代理/第二身份
}
else{
scanf("%d",&a);
majia[a]=cnt++; //改变它的代理/第二身份。
}
}
for(int i=;i<n;i++){
int v=find_root(majia[i]);
if(!vis[v]){
ans++;
vis[v]=;
}
} //printf("Case #%d: %d\n",t,group.size());
printf("Case #%d: %d\n",t,ans);
} return ;
}

HDU 2473 Junk-Mail Filter(并查集+删点,设立虚父节点/找个代理)的更多相关文章

  1. (step5.1.2)hdu 2473(Junk-Mail Filter——并查集)

    题目大意:输入两个整数n,m(n表示点的个数,m表示操作数).在接下来的m行中,对点的操作有两种 1)M a b . 表示将a.b并到一个集合中 2)S a .表示将a从原来的集合中去除,而成为一个单 ...

  2. HDU——2473Junk-Mail Filter(并查集删点)

    Junk-Mail Filter Time Limit: 15000/8000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) T ...

  3. Junk-Mail Filter 【并查集虚父节点】

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2473 题目大意: n个点,m个操作,操作时,输入M a b,表示a, b在一个集合里, 输入S a 表 ...

  4. ZOJ 3261 - Connections in Galaxy War ,并查集删边

    In order to strengthen the defense ability, many stars in galaxy allied together and built many bidi ...

  5. hdu 2473 Junk-Mail Filter (并查集之点的删除)

    Junk-Mail Filter Time Limit: 15000/8000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  6. HDU 2473 Junk-Mail Filter 并查集,虚拟删除操作

    http://acm.hdu.edu.cn/showproblem.php?pid=2473 给定两种操作 第一种是合并X Y 第二种是把X分离出来,就是从原来的集合中分离出来,其它的关系不变. 关键 ...

  7. HDU 2473 Junk-Mail Filter(并查集的删除操作)

    题目地址:pid=2473">HDU 2473 这题曾经碰到过,没做出来. .如今又做了做,还是没做出来. ... 这题涉及到并查集的删除操作.想到了设一个虚节点,可是我把虚节点设为了 ...

  8. HDU 2473 Junk-Mail Filter 并查集删除(FZU 2155盟国)

    http://acm.hdu.edu.cn/showproblem.php?pid=2473 http://acm.fzu.edu.cn/problem.php?pid=2155 题目大意: 编号0~ ...

  9. hdu2473 Junk-Mail Filter 并查集+删除节点+路径压缩

    Description Recognizing junk mails is a tough task. The method used here consists of two steps:  1) ...

随机推荐

  1. Linux读写锁的使用

    读写锁是用来解决读者写者问题的,读操作可以共享,写操作是排它的,读可以有多个在读,写只有唯一个在写,写的时候不允许读. 具有强读者同步和强写者同步两种形式: 强读者同步:当写者没有进行写操作时,读者就 ...

  2. Linux下OpenCV的环境搭建

    OpenCV is the most popular and advanced code library for Computer Vision related applications today, ...

  3. Python单元测试——深入理解unittest (转)

    单元测试的重要性就不多说了,可恶的是Python中 有太多的单元测试框架和工具,什么unittest, testtools, subunit, coverage, testrepository, no ...

  4. linux 安装sysstat使用iostat、mpstat、sar、sa(转载)

    使用yum安装 #yum install sysstat sysstat的安装包是:sysstat-5.0.5-1.i386.rpm,装完了sysstat-5.0.5-1.i386.rpm后 就会有i ...

  5. WPF 绑定二(绑定指定的字符串)

    xaml: <Window x:Class="WpfApplication1.Window2" xmlns="http://schemas.microsoft.co ...

  6. “~/Views/Home/Text.aspx”处的视图必须派生自 ViewPage、ViewPage<TModel>、ViewUserControl 或 ViewUserControl<TModel>。

    在MVC架构中使用aspx页面,需要在Text.aspx中开头加入如下代码: <%@ Page Language="C#" Inherits="System.Web ...

  7. CSS各个浏览器Hack的写法

    Hack是针对不同的浏览器去写不同的CSS样式,从而让各浏览器能达到一致的渲染效果,那么针对不同的浏览器写不同的CSS CODE的过程,就叫CSS HACK,同时也叫写CSS Hack.然后将Hack ...

  8. Posix 共享内存区

    要点 与mmap配合使用 open与shm_open的区别,open打开磁盘上的普通文件,shm_open创建和打开的文件在/dev/shm文件夹下,该文件夹对应的是内存 gcc编译时加参数-lrt ...

  9. 敏捷开发之道(四)Scrum概述

    上次的博文敏捷开发之道(二)极限编程XP和敏捷开发之道(三)极限编程XP续中,我们介绍了一下敏捷开发中的XP开发方法,今天咱们来了解另一个比较流行的敏捷开发方法--Scrum. 1.Scrum简介 S ...

  10. 你用什么方法检查 PHP 脚本的执行效率(通常是脚本执行时间)和数据库 SQL 的效率(通常是数据库 Query 时间), 并定位和分析脚本执行和数据库查询的瓶颈所在?

    php: 一般是在你要检查的代码开头记录一个时间,结尾记录一个时间.取差值, 数据库SQL的效率    sql的explain(mysql),启用slow query log记录慢查询.   通常还要 ...