首页 > 其他分享 >题解_P2024 [NOI2001] 食物链

题解_P2024 [NOI2001] 食物链

时间:2024-07-31 16:19:11浏览次数:13  
标签:P2024 int 题解 find ++ NOI2001 ans eat op

[NOI2001] 食物链

题目描述

动物王国中有三类动物 \(A,B,C\),这三类动物的食物链构成了有趣的环形。\(A\) 吃 \(B\),\(B\) 吃 \(C\),\(C\) 吃 \(A\)。

现有 \(N\) 个动物,以 \(1 \sim 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\) 句话,输出假话的总数。

输入格式

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

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

输出格式

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

样例 #1

样例输入 #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\le N\le 5 \times 10^4\),\(1\le K \le 10^5\)。

题解

思路分析

题目中涉及到需要快速得出ABC三类之间的关系,那么明显可以考虑并查集。

【方法1】带权并查集
d[u] 表示 x 与其对应根节点 root 的关系,0 表示同类,1 表示吃,2 表示被吃。

int find(int u) {
    if (u != p[u]) {
        int fa = p[u];
        p[u] = find(p[u]);
        d[u] = (d[u] + d[fa]) % 3;
    }
    return p[u];
}

// (x,y) 同类合并
int a = find(x), b=find(y);
d[x] + d[a] = d[y]. --> d[a] = (d[y] - d[x] + 3) % 3;

// (x eat y) 合并
d[x] + d[a] = d[y] + 1. --> d[a] = (d[y] - d[x] + 4) % 3;

【方法1】扩展域并查集
将 p[] 数组扩展为原数组的 3 倍,p[1~n] 为 A 类,p[n+1~n2] 为 B 类,p[n2+1~n*3] 为 C 类。

也就是说 (u eat u+n), (u+n eat u+n2) , (u+n2 eat u)


// (x,y) 同类合并
p[find(x)] = find(y);
p[find(x + n)] = find(y + n);
p[find(x + n * 2)] = find(y + n * 2);

// (x eat y) 合并
p[find(x)] = find(y + n * 2);
p[find(x + n)] = find(y);
p[find(x + n * 2)] = find(y + n);

程序实现

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 5e4 + 10, INF = 0x3f3f3f3f;
int n, k, p[N * 3], d[N * 3];

int find(int u) {
    if (u != p[u]) {
        int fa = p[u];
        p[u] = find(p[u]);
        d[u] = (d[u] + d[fa]) % 3;
    }
    return p[u];
}
// 带权并查集
int solve1() {
    int op, x, y, ans = 0;
    cin >> n >> k;
    for (int i = 0; i <= n; i++) p[i] = i, d[i] = 0;
    while (k--) {
        cin >> op >> x >> y;
        if (x > n || y > n) { ans++; continue; }
        int a = find(x), b = find(y);
        if (op == 1) {  // x-y 同类
            if (a == b && (d[x] - d[y] + 3) % 3) ans++;
            if (a != b) {
                p[a] = b;
                d[a] = (d[y] - d[x] + 3) % 3;
            }
        } else {  // x eat y
            if (a == b && (d[x] - d[y] + 3) % 3 != 1) ans++;
            if (a != b) {
                p[a] = b;
                d[a] = (d[y] - d[x] + 4) % 3;
            }
        } 
    }
    cout << ans << endl;
    return 0;
}

// 扩展域并查集
int solve2() {
    int op, x, y, ans = 0;
    cin >> n >> k;
    for (int i = 0; i <= n * 3; i++) p[i] = i;
    while (k--) {
        cin >> op >> x >> y;
        if (x > n || y > n) { ans++; continue; }
        if (op == 1) {  // x-y 同类
            if (find(x) == find(y + n) || find(y) == find(x + n)) {
                ans++; continue;
            }
            p[find(x)] = find(y);
            p[find(x + n)] = find(y + n);
            p[find(x + n * 2)] = find(y + n * 2);
        } else {  // X eat y   (x 吃 x+n)
            if (find(x) == find(y) || find(x) == find(y + n)) {  
                ans++; continue;
            }
            p[find(x)] = find(y + n * 2);
            p[find(x + n)] = find(y);
            p[find(x + n * 2)] = find(y + n);
        }
    }
    cout << ans << endl;
    return 0;
}
int main() {
    solve1();
    // solve2();
}

标签:P2024,int,题解,find,++,NOI2001,ans,eat,op
From: https://www.cnblogs.com/hellohebin/p/18334891

相关文章

  • 20240731题解
    这么简单的题目没有AK(计时器(timer)题目:每次可以加上\(2^n-1\),问多少次变成\(x\)题解:因为较大的数大于较小的数的两倍,直接贪心的选最大的即可。复杂度\(\Theta(T\logn)\)代码:#include<cstdio>#defineintlonglongconstintN=105,A=1000000000000000000;intT,x,f[N......
  • P4784 城市 题解 / 最小斯坦纳树
    P4784城市题解题目大意给定\(n\)个节点,\(m\)条带边权边,和\(k\)重要节点。选择一些边,使得这些边能让这\(k\)个节点连通,代价为选出的边权和。求最小代价。(\(k\leq5\))Solve前置芝士:斯坦纳树。定义将指定点集合(部分点)中的所有点连通,且边权总和最小的生成树称为最小斯坦......
  • P3501 [POI2010] ANT-Antisymmetry 反对称 题解(字符串哈希+二分)
    原题题意若一个由010101组成的字符串将000和......
  • ARC180 部分简要题解
    C设\(f_{i,j}\)为考虑前\(i\)个数,当前选出来的子序列和为\(j\)且强制最后一个选出来的数不为\(j\)的方案;设\(g_{i,j}\)为考虑前\(i\)个数,当前选出来的子序列和为\(j\)且强制最后一个选出来的数必为\(j\)的方案。注意到一个合法方案可以唯一与一个最后一个选出......
  • P10814 【模板】离线二维数点 题解
    题目传送门思路一眼主席树板子题,但是一看数据范围\(n,m\le2\times10^6\),似了。在线做法应该是似完了,考虑离线做法。我们知道树状数组是可以做二维偏序的,大家应该都知道一个经典问题:对于一个序列,多次询问下标\(\lea\)且数值\(\leb\)的数的个数。回到这道题,相比上面......
  • P2163 [SHOI2007] 园丁的烦恼 题解
    题目传送门题目大意:在一个平面直角坐标系上,给定\(n\)个点的坐标\((x,y)\),\(m\)次询问,每次询问一个矩形范围内的点的数量,此矩形用\(\{a,b,c,d\}\)来描述,其中\((a,b)\)为左下角,\((c,d)\)为右上角。思路:不难将题目转化为:给定一个长度为\(n\)的序列,序列中的每个元......
  • CF1997(edu168)题解 A-F
    A.StrongPassword注意到最大效果是在两个相同字符之间插入一个不同的,贡献为3。否则在一开始插入一个和首位不同的,贡献为2。#include<bits/stdc++.h>usingnamespacestd;typedeflonglongll;voidsolve(){strings;cin>>s;boolok=0;for(inti......
  • 【题解】2024牛客多校第5场
    E安https://ac.nowcoder.com/acm/contest/81600/E分析简单博弈/思维题。当ai>bi时,当前骑士一定存活。当ai<bi时,当前骑士一定死亡。为了使得自己存活的骑士尽可能多,若存在ai=bi的情况,一定会选择令该骑士去攻击对方,并且双方均会轮流优先选择此类骑士。......
  • 07-30 题解
    07-30题解A朴素的想法$dp(i,j,k)$表示考虑到第\(i\)位,前\(i\)位的和为\(j\),第\(i\)位的值为\(k\)然后咋转移?重新定义移动小球的方式:从自己右边的邻居拿过来正数个球拿过来负数个球(即往右边的邻居放正数个球)在第2种操作中,我们拿走的球会被后面放过来......
  • Luogu P1983 车站分级 题解 [ 绿 ] [ 拓扑排序 ] [ 图论建模 ] [ 虚点 ]
    车站分级:很好的拓扑排序题,细节有点多。图论建模首先观察对于一条线路,我们可以从中直接得到什么信息:假设这条线路的开头为\(st\),结尾为\(ed\),那么在\([st,ed]\)的车站中,没有被选入线路的点一定比选入线路的点的级数至少少\(1\)。对于这一点条件,我们就可以建模了。......