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 ...
随机推荐
- P3868 [TJOI2009]猜数字
[TJOI2009]猜数字 中国剩余定理 求解i=1 to n : x≡a[i] (mod b[i])的同余方程组 设 t= ∏i=1 to n b[i] 我们先求出 i=1 to n : x≡1 ( ...
- Python3 实现(wxpy)用微信自动定时给朋友定时推广
Python3 实现(wxpy)用微信自动定时给朋友定时推广 安装wxpy 安装微信机器人wxpy非常简单,如果你拥有pip,请直接按照Github中的方法安装: pip install wxpy 实 ...
- HTML5 Chart.js 框架
HTML5 Chart.js 框架 版权声明:未经博主授权,内容严禁转载 ! Chart.js 概述: chart.js 是一个简单的.面向对象.为设计者开发者准备的图表绘制工具. 点击进入官方网址 ...
- shell分析http日志
http状态码1字头----信息,服务器收到请求,需要请求者继续执行操作2字头----成功,操作被成功接收并处理3字头----重定向,需要进一步的操作以完成请求4字头----客户端错误,请求包含语法错 ...
- 详解 HTML5 中的 WebSocket 及实例代码-做弹幕
原文链接:http://www.php.cn/html5-tutorial-363345.html
- 《网络攻防》实验八:Web基础
适逢多事之际,下周二的课设答辩.全国信安竞赛初赛作品筹备.协会密码沙龙比肩接踵,这些"案牍"不仅劳形还影响了我的复习计划."甘蔗没有两头甜的"还是要有所舍得了, ...
- 如何高效判断java数组是否包含某个值
在java中,我们如何判断一个未排序数组中是否包含一个特定的值?这在java中是一个频繁非常实用的操作.那么什么样的方法才是最高效的方式?当然 ,这个问题在Stack Overflow也是得票率非常高 ...
- python3+pyqt5 +eric5安装配置
一.大纲内容: 1.预备PC环境: 2.预备安装程序: 2.1.下载Python3.2 2.2.下载PyQt4 2.3.下载Eric5 3.安装配置步骤: 3.1.安装Pyhon3.2 3.2.安装P ...
- The equation (扩展欧几里得)题解
There is an equation ax + by + c = 0. Given a,b,c,x1,x2,y1,y2 you must determine, how many integer r ...
- JVM类加载机制总结
1.运行时加载优点 提高灵活性,可以在运行时动态加载,连接.例子:面向接口编程,动态绑定实现类(但C++也有动态绑定,说明动态绑定不一定通过运行时加载Class字节码实现,也可能是机器码支持的) 2. ...