洛谷P2024 [NOI2001]食物链(带权并查集,扩展域并查集)

2024-04-16 02:18

本文主要是介绍洛谷P2024 [NOI2001]食物链(带权并查集,扩展域并查集),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目描述
动物王国中有三类动物 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 句话有的是真

的,有的是假的。当一句话满足下列三条之一时,这句话就是假话,否则就是真话。

• 当前的话与前面的某些真的话冲突,就是假话

• 当前的话中 X 或 Y 比 N 大,就是假话

• 当前的话表示 X 吃 X,就是假话

你的任务是根据给定的 N 和 K 句话,输出假话的总数。

输入格式
从 eat.in 中输入数据

第一行两个整数,N,K,表示有 N 个动物,K 句话。

第二行开始每行一句话(按照题目要求,见样例)

输出格式
输出到 eat.out 中

一行,一个整数,表示假话的总数。

输入输出样例
输入 #1复制
100 7
1 101 1
2 1 2
2 2 3
2 3 3
1 1 3
2 3 1
1 5 5
输出 #1复制
3
说明/提示
1 ≤ N ≤ 5 ∗ 10^4

1 ≤ K ≤ 10^5

图片来源自:https://www.cnblogs.com/zhanyage110/p/4394716.html
在这里插入图片描述
在这里插入图片描述

思路: 用两种写法回顾了一下种类并查集,两者等价,都是关系的转移

带权并查集写法:
我们定义 d [ x ] d[x] d[x] x x x与根节点的关系,
x=0代表与根节点同类
x=1代表吃根节点
x=2代表被根节点吃

题中给出的关系减一就等于 d [ x ] d[x] d[x]对应的关系,然后这样维护就好了。

向量偏移
ACNEW

#include<iostream>
#include<cstdio>
#include<vector>
#include<unordered_set>
using namespace std;const int maxn = 5e4 + 7;int fa[maxn],d[maxn];int findset(int x) {if(x == fa[x]) {return x;}int rt = findset(fa[x]);d[x] = (d[fa[x]] + d[x]) % 3;return fa[x] = rt;
}int main() {int n,m;scanf("%d%d",&n,&m);for(int i = 1;i <= n;i++) {fa[i] = i;}int res = 0;for(int i = 1;i <= m;i++) {int x,y,z;scanf("%d%d%d",&z,&x,&y);if(x > n || y > n) {res++;continue;}if(z == 2 && x == y) {res++;continue;}int rx = findset(x),ry = findset(y);if(rx == ry) {if((d[x] - d[y] + 3) % 3 != (z - 1 + 3) % 3) {res++;}} else {fa[rx] = ry;d[rx] = (z - 1 + d[y] - d[x] + 3) % 3;}}printf("%d\n",res);return 0;
}

扩展域写法:
对于x,
x为同类域
x+n为捕食域
x+2*n为天敌域

扩展域就是对每个数维护多个关系区间

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;const int maxn = 3e5 + 7;int fa[maxn];int findset(int x) {if(x == fa[x]) {return x;}return fa[x] = findset(fa[x]);
}void Union(int x,int y) {int rx = findset(x),ry = findset(y);if(rx != ry) {fa[rx] = ry;}
}int main() {int n,m;scanf("%d%d",&n,&m);int res = 0;for(int i = 1;i <= 3 * n;i++) {fa[i] = i;}for(int i = 1;i <= m;i++) {int x,y,z;scanf("%d%d%d",&z,&x,&y);if(z == 2 && x == y) {res++;continue;}if(x > n || y > n) {res++;continue;}if(z == 1 && (findset(x) == findset(y + n) || findset(x) == findset(y + 2 * n))) { //x吃y和y吃x都不行res++;continue;}if(z == 2 && (findset(x) == findset(y) || findset(x) == findset(y + n))) {res++;continue;//x和y同类和x被y吃都不行}if(z == 1) {Union(x,y);Union(x + n,y + n);Union(x + 2 * n,y + 2 * n);} else {Union(x,y + 2 * n);Union(x + n,y);Union(x + 2 * n,y + n);}}printf("%d\n",res);return 0;
}

这篇关于洛谷P2024 [NOI2001]食物链(带权并查集,扩展域并查集)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/907552

相关文章

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

poj 1182 并查集 食物链类

题意: 有n只动物,分别编号1....n。所有动物都属于A,B,C中的一种,已知A吃B,B吃C,C吃A。 按顺序给出下面两种共K条信息: 1. x 和 y 属于同一类。 2. x 吃 y 。 然而这些信息可能会出错,有可能有的信息和之前给出的信息矛盾,也有的信息可能给出的 x 和 y 不在n的范围内。 求k条信息中有多少条是不正确的。 解析: 对于每只动物,创建3个元素 i

Spring框架5 - 容器的扩展功能 (ApplicationContext)

private static ApplicationContext applicationContext;static {applicationContext = new ClassPathXmlApplicationContext("bean.xml");} BeanFactory的功能扩展类ApplicationContext进行深度的分析。ApplicationConext与 BeanF

POJ1988带权并查集

M u v 将u所在的堆移动到v所在的堆的上面 C u 求u的下面有多少块 带权并查集 import java.io.BufferedReader;import java.io.InputStream;import java.io.InputStreamReader;import java.io.PrintWriter;import java.math.BigInteger;i

POJ1703带权并查集

D: u v  u与v在不同的集合 A: u v  查询u与v的关系 1)压缩路径过程        fu->root   0  1  u-fu 0                 0  1   1                 1  0 2)合并过程 fu->fv      u->fu   0 1   v->fv 0            1 0 1            0

PHP7扩展开发之数组处理

前言 这次,我们将演示如何在PHP扩展中如何对数组进行处理。要实现的PHP代码如下: <?phpfunction array_concat ($arr, $prefix) {foreach($arr as $key => $val) {if (isset($prefix[$key]) && is_string($val) && is_string($prefix[$key])) {$arr[

PHP7扩展开发之字符串处理

前言 这次,我们来看看字符串在PHP扩展里面如何处理。 示例代码如下: <?phpfunction str_concat($prefix, $string) {$len = strlen($prefix);$substr = substr($string, 0, $len);if ($substr != $prefix) {return $prefix." ".$string;} else

PHP7扩展开发之类型处理

前言 这次,我们将演示如何在PHP扩展中如何对类型进行一些操作。如,判断变量类型。要实现的PHP代码如下: <?phpfunction get_size ($value) {if (is_string($value)) {return "string size is ". strlen($value);} else if (is_array($value)) {return "array si

PHP7扩展开发之依赖其他扩展

前言 有的时候,我们的扩展要依赖其他扩展。比如,我们PHP的mysqli扩展就依赖mysqlnd扩展。这中情况下,我们怎么使用其他扩展呢?这个就是本文讲述的内容。 我们新建立一个扩展,名字叫 demo_dep , 依赖之前的say扩展。 在demo_dep扩展中,我们实现demo_say方法。这个方法调用say扩展的say方法。 代码 基础代码 确保say扩展的头文件正确安装到了php