POJ 1182 食物链(并查集+偏移向量)题解

时间:2022-03-13 09:24:11
食物链
Time Limit: 1000MS   Memory Limit: 10000K
Total Submissions: 82346   Accepted: 24616

Description

动物王国中有三类动物A,B,C,这三类动物的食物链构成了有趣的环形。A吃B, B吃C,C吃A。 
现有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

第一行是两个整数N和K,以一个空格分隔。 
以下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

思路:

因为要表达并查集间的关系,所以我们设了一组数组rela[i]来表示i与他的根之间的关系(即表示i到pre[i]的偏移量),假设a b,0表示同类,1表示a吃b,2表示b吃a。

在压缩路径时,我们需要对沿途的数据更新rela:rela[a]=(rela[a]+rela[ pre[a] ])%3;  此代码应这样理解:rela表示的是a对当前的根aa的偏移量,即a->aa,而aa对最顶端的根fa偏移量为rela[ aa ],即aa->fa,所以将两者相加(注意!这是个向量,即a->根)就为a对最顶端的根fa偏移量,即a->fa。

在执行join时我们注意到我们需要更新合并后fb的rela,代码为:rela[fb]=(3-rela[b]+(c-1)+rela[a])%3;  要了解这个,首先我们要知道一个知识:

假如x作为y的根时,y的偏移量是a(即y->x),那么y作为x的父亲时,x的偏移量就是(3 - a)%3(即x->y)

所以代码应这样理解:(3-rela[b])是fb对b的偏移量即fb->b, (c-1)是b对a偏移量即b->a,(rela[a]) 是a对fa的原根偏移量

即a->fa ,所以fb->b +b->a + a->fa =fb->fa,即为fb对fa偏移量。

我也是在看了大佬详细解说之后才恍然大悟,特奉上参考博客:点此

代码:

#include<stdio.h>
#include<string.h>
#include<math.h>
#include<stdlib.h>
#include<map>
#define N 50010
#define HASH 10000
using namespace std;
int pre[N],rela[N],c; int find(int a){
if(pre[a]==a) return a;
else{
int temp=pre[a];
pre[a]=find(temp);
rela[a]=(rela[a]+rela[temp])%3; //当前a对根的偏移量
return pre[a];
}
}
void join(int a,int b){
int fa=find(a);
int fb=find(b);
if(fa!=fb){
pre[fb]=fa;
rela[fb]=(3-rela[b]+(c-1)+rela[a])%3;//(3-rela[b])是b与原根fb的偏移量, (c-1)是a b间偏移量,(rela[a]) 是a和它的原根fa偏移量 ,三者相加为fa fb间偏移量
}
}
int main(){
int n,i,k,answer=0,a,b;
scanf("%d%d",&n,&k);
for(i=0;i<=n;i++){
pre[i]=i;
rela[i]=0;
}
while(k--){
scanf("%d%d%d",&c,&a,&b);
if(c==2 && a==b) answer++;
else if(a>n ||b>n) answer++;
else if(find(a)!=find(b)){
join(a,b);
}
else if(c==1){
if(rela[a]!=rela[b])
answer++;
}
else{
if( ((rela[b]+3-rela[a])%3)!=1 ) //代表a为b根时b的偏移量
answer++;
}
}
printf("%d\n",answer);
return 0;
}

POJ 1182 食物链(并查集+偏移向量)题解的更多相关文章

  1. POJ 1182 食物链 &lbrack;并查集 带权并查集 开拓思路&rsqb;

    传送门 P - 食物链 Time Limit:1000MS     Memory Limit:10000KB     64bit IO Format:%I64d & %I64u Submit  ...

  2. poj 1182 食物链 并查集 题解《挑战程序设计竞赛》

    地址 http://poj.org/problem?id=1182 题解 可以考虑使用并查集解决 但是并不是简单的记录是否同一组的这般使用 每个动物都有三个并查集 自己 天敌 捕食 并查集 那么在获得 ...

  3. POJ 1182 食物链&lpar;并查集的使用)

    食物链 Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 81915   Accepted: 24462 Description ...

  4. poj 1182 食物链 并查集的又一个用法

    食物链   Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 41584   Accepted: 12090 Descripti ...

  5. POJ 1182食物链&lpar;并查集)

    食物链Time Limit: 1000MS Memory Limit: 10000KTotal Submissions: 85474 Accepted: 25549Description动物王国中有三 ...

  6. POJ - 1182 食物链 并查集经典

    思路:设r(x)表示节点x与根结点的关系,px表示x的根结点.记录每个节点与其父节点的关系,就能很方便知道每个节点以及和它的父节点的关系. struct node{ int par; //父亲节点 i ...

  7. poj——1182食物链 并查集(提升版)

    因为是中文题,题意就不说了,直接说思路: 我们不知道给的说法中的动物属于A B C哪一类,所以我们可以用不同区间的数字表示这几类动物,这并不影响结果,我们可以用并查集把属于一类的动物放在一块,举个例子 ...

  8. POJ 1182 食物链 &lpar;并查集&rpar;

    食物链 Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 50601   Accepted: 14786 Description ...

  9. POJ 1182 食物链&lpar;并查集)

    题目链接 经过宝哥的讲解,终于对这种问题有了进一步的理解.根据flag[x]和flag[y]求flag[tx]是最关键的了. 0吃1,1吃2,2吃0. 假设flag[tx] = X; 那么X + fl ...

随机推荐

  1. kail linux 虚拟机安装实录(一) 新建虚拟机

    各位晚上好. 现在开始进行kail linux 在虚拟机上的安装. 我所使用的工具如下: kail linux 2.0   x64  http://mirrors.neusoft.edu.cn/kal ...

  2. jeecg安装——mysql数据库创建&plus;手动执行初始化脚本

    国产的开源项目官方文档写得那么详细,已经是很厚道了,可惜俺这种菜鸟还是会碰到各种"小"问题,做个笔记先! 1.新建数据库: CREATE DATABASE jeecg DEFAUL ...

  3. 关于 实时推送技术--WebSocket的 知识分享

    今天学习了关于WebSocket的知识,觉得挺有用的,在这记录一下,也和大家分享一下!!有兴趣的可以看看哦 WebSocket简介 Web领域的实时推送技术,也被称作Realtime技术.这种技术要达 ...

  4. js&comma;获取和设置cookie、 localStorage

    cookie 跟域名有关系的浏览器缓存 获取cookie document.cookie; 获取到的如果是多条cookie信息,是以分号和一个空格("; ")隔开:a=1; b=2 ...

  5. appium 报错

    2018-11-27 18:05:56:313 - [Logcat] Logcat terminated with code 0, signal null 2018-11-27 18:05:56:31 ...

  6. 嵌入式开发之hi3519---GPIO 按键驱动

    摸索了一个星期,终于把海思HI3515开发板的按键中断程序搞出来了,hi3515的核心芯片与网上例子较多的s3c之类的有一些区别,以至于浪费了好些时间去琢磨.管脚配置方式不一样,中断的使用情况也不一样 ...

  7. Microsoft SQL Server on Linux 踩坑指南

    微软用 SQL Server 在 2016 年的时候搞了一个大新闻,宣传 Microsoft ❤️ Linux 打得一众软粉措手不及.但是这还是好事情,Linux 上也有好用的 SQL Server ...

  8. greenDao 介绍

    greenDAO是一个针对Android的轻快速ORM解决方案,它将对象映射到SQLite数据库.http://greenrobot.org/greendao/ greenDAO is a light ...

  9. &lbrack;QA翻译&rsqb;如何在Storm里拆分stream流?

    原文:http://*.com/questions/19807395/how-would-i-split-a-stream-in-apache-storm 问题:我现在不清楚如 ...

  10. 【转载】利用MAVEN打包时,如何包含更多的资源文件

    首先,来看下MAVENx项目标准的目录结构: 一般情况下,我们用到的资源文件(各种xml,properites,xsd文件等)都放在src/main/resources下面,利用maven打包时,ma ...