poj 1182 (带权并查集)
| Time Limit: 1000MS | Memory Limit: 10000K | |
| Total Submissions: 71361 | Accepted: 21131 |
Description
现有N个动物,以1-N编号。每个动物都是A,B,C中的一种,但是我们并不知道它到底是哪一种。
有人用两种说法对这N个动物所构成的食物链关系进行描述:
第一种说法是"1 X Y",表示X和Y是同类。
第二种说法是"2 X Y",表示X吃Y。
此人对N个动物,用上述两种说法,一句接一句地说出K句话,这K句话有的是真的,有的是假的。当一句话满足下列三条之一时,这句话就是假话,否则就是真话。
1) 当前的话与前面的某些真的话冲突,就是假话;
2) 当前的话中X或Y比N大,就是假话;
3) 当前的话表示X吃X,就是假话。
你的任务是根据给定的N(1 <= N <= 50,000)和K句话(0 <= K <= 100,000),输出假话的总数。
Input
以下K行每行是三个正整数 D,X,Y,两数之间用一个空格隔开,其中D表示说法的种类。
若D=1,则表示X和Y是同类。
若D=2,则表示X吃Y。
Output
Sample Input
100 7
1 101 1
2 1 2
2 2 3
2 3 3
1 1 3
2 3 1
1 5 5
Sample Output
3
欸困惑许久的一道题,寒假学并查集时已经遇见,卡在带权并查集一直没时间or感觉很复杂就没学,前几日突然想起,便想起他了.
带权并查集感觉在于对向量偏移的理解和对getf()函数递归查找祖先的深刻理解,怎奈数学砸砸看见数学就头疼>_<
本题体面给出了三个类,并查集的作用:将具有相同性质的某些点归到一颗树里,他们有共同的祖先.
难不成三个类要用三个并查集?显然用三个只会更加复杂,我们不妨利用带权并查集点到祖先的距离的不同来表示点与祖先的关系,将所有的点归到一颗树下,假设有两个点x,y在一颗树里且我们知道其到祖先的距离d[x],d[y],
显然利用这两个"距离"(关系)我们足以推出x,y的关系,这便涉及到向量了.
带权并查集主要就是在getf()函数递归的过程中顺便改变到根节点的距离这一步!合并函数中也需要进行小小的改变.
推论:已知x,x的父亲t=f[x],根祖先root和d[x],d[t]; 我们要做的就是根据 d[x],d[t]更新出当前正确的d[x]的值.
穷举出所有的可能后:{0,1,2}--{0,1,2}不难得出结论 d[x]=(d[x]+d[t])%3;
为了保证 d[t]的值是x的父亲到祖先的距离我们需要先递归x的父亲更新他父亲的距离......
如果不先进行递归的话,d[t]的值可能并不是t到其根的距离导致结果错误!
merg函数类似利用向量推出公式即可,利用0,1,2的巧妙之处在于可以用(d-1)直接表示出x与y的关系.
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int f[50005],d[50005];
int getf(int v)
{
if(f[v]==v) return v;
int t=f[v];
f[v]=getf(f[v]);
d[v]=(d[v]+d[t])%3;
return f[v];
}
void merg(int x,int y,int rx,int ry,int D)
{
f[ry]=rx;
d[ry]=(3-d[y]+(D-1)+d[x])%3;
}
int main()
{
int n,k,x,y,D,ans=0,i,j;
cin>>n>>k;
for(i=1;i<=n;++i) f[i]=i;
memset(d,0,sizeof(d));
while(k--){
scanf("%d%d%d",&D,&x,&y);
if(x>n||y>n||(x==y&&D==2)) {++ans;continue;}
int rx=getf(x),ry=getf(y);
if(rx==ry){
if((D-1)!=(3-d[x]+d[y])%3) ++ans;
}
else merg(x,y,rx,ry,D);
}
cout<<ans<<endl;
return 0;
}
poj 1182 (带权并查集)的更多相关文章
- poj 1733(带权并查集+离散化)
题目链接:http://poj.org/problem?id=1733 思路:这题一看就想到要用并查集做了,不过一看数据这么大,感觉有点棘手,其实,我们仔细一想可以发现,我们需要记录的是出现过的节点到 ...
- POJ 1703 带权并查集
直接解释输入了: 第一行cases. 然后是n和m代表有n个人,m个操作 给你两个空的集合 每个操作后面跟着俩数 D操作是说这俩数不在一个集合里. A操作问这俩数什么关系 不能确定:输出Not sur ...
- Navigation Nightmare POJ - 1984 带权并查集
#include<iostream> #include<cmath> #include<algorithm> using namespace std; ; // 东 ...
- Parity game POJ - 1733 带权并查集
#include<iostream> #include<algorithm> #include<cstdio> using namespace std; <& ...
- POJ 1182 食物链 [并查集 带权并查集 开拓思路]
传送门 P - 食物链 Time Limit:1000MS Memory Limit:10000KB 64bit IO Format:%I64d & %I64u Submit ...
- POJ 1182 食物链 【带权并查集】
<题目链接> 题目大意: 动物王国中有三类动物A,B,C,这三类动物的食物链构成了有趣的环形.A吃B, B吃C,C吃A. 现有N个动物,以1-N编号.每个动物都是A,B,C中的一种,但是我 ...
- POJ 1182 食物链 (带权并查集)
食物链 Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 78551 Accepted: 23406 Description ...
- POJ 1182 食物链 【带权并查集/补集法】
动物王国中有三类动物A,B,C,这三类动物的食物链构成了有趣的环形.A吃B, B吃C,C吃A. 现有N个动物,以1-N编号.每个动物都是A,B,C中的一种,但是我们并不知道它到底是哪一种.有人用两种说 ...
- poj 1182 食物链【带权并查集】
设相等的边权为0,吃的边权为,被吃的边权为2,然后用带权并查集在%3的意义下做加法即可 关系为简单环的基本都可以用模环长的方式是用带权并查集 #include<iostream> #inc ...
随机推荐
- 出现“基础链接已关闭,无法链接到远程服务器"错误的解决办法
一些用户在安装一些软件或是系统做某些修改后,采集器就没无登录或是无法获取到网页.登录或是使用httppostget工具会出现 ”基础链接已关闭,无法链接到远程服务器“的提示.经分析,是系统Socket ...
- jdk自带的ThreadLocal和netty扩展的FastThreadLocal比较总结
最近在分析一潜在内存泄露问题的时候,jmap出来中有很多的FastThreadLocalThread实例,看了下javadoc,如下: A special variant of ThreadLocal ...
- Python descriptor 以及 内置property()函数
Python Descriptor 1, Python Descriptor是这样一个对象 它按照descriptor协议, 有这样的属性之一 def __get__(self, obj, type ...
- 自动对比度的opencv实现
在http://www.cnblogs.com/Imageshop/archive/2011/11/13/2247614.html 一文中,作者给出了“自动对比度”的实现方法,非常nice 实际实现过 ...
- 20145336《网络对抗技术》Exp6 信息搜集技术
20145336张子扬 <网络对抗技术> 信息搜集与漏洞扫描 实验内容 使用whois进行域名注册信息查询,使用nslookup进行域名查询 实现对IP地理位置的查询 使用PING.nam ...
- bzoj 3600 没有人的算术 - 替罪羊树 - 线段树
题目都是图片,就不给了,就给链接好了 由于bzoj比较慢,就先给[vjudge传送门] 有兴趣的可以去逛bzoj[bzoj传送门] 题目大意 有n个数a[1],a[2],...,a[n],它们开始都是 ...
- JAVA I/O(四)网络Socket和ServerSocket
<Thinking in Enterprise Java>中第一章描述了用Socket和Channel的网络编程,核心即为Socket和Channel,本文简单讲述Socket的应用. S ...
- Linux命令中:rsync和cp之间的区别
rsync:只拷贝那些更新的文件: cp -u:也可以实现类似效果: 两者都基本可以满足备份的需求: 只是一般情况下,用rsync做这类备份之类的事情,更多见: 在备份的操作中,拷贝,过期文件的删除是 ...
- windows的gvim总是报错: +iconv fencview.vim
iconv是用来转换gvim文件的编码的, 需要插件: iconv.dll gvim7.3的文件目录结构: vim/vim73是它的核心文件, 而vimfiles是扩展文件, 里面的plugin是专门 ...
- Leetcode——Target Sum
Question You are given a list of non-negative integers, a1, a2, ..., an, and a target, S. Now you ha ...