Luogu P9869 NOIp2023 三值逻辑 题解 [ 绿 ] [ 带权并查集 ]
三值逻辑:有点坑并且细节较繁琐,但有点板子的并查集。
修改操作
发现对于每个点,只有对他的最后一次操作才是有用的,所以记录下最终的祖先即可。
然而这里并不能用并查集来实现,因为并查集它具有的是传递性,无论你路不路径压缩,每次修改一个父节点时它的子节点一定会被修改,所以我们不能使用并查集。
但是可以使用并查集相关思想。
首先我们建立虚拟源点 \(n+1\) 表示 \(T\),\(n+2\) 表示 \(U\)。到某个点的距离为奇数时表示与这个点的值相反,为偶数则表示与这个点的值相等。
每次修改后,我们都直接指向父节点的祖先节点。为啥祖先节点就可以呢?因为此时祖先节点是未被修改的,而根据题目中“使得每个变量初始值与最终值相同”,所以这里跟祖先节点连边相当于和祖先节点的初值连边。自然是正确的了。
upd:这里其实可以用并查集,但是要时时刻刻路径压缩,每 combine 一次就要压一次。
注意,本题的先后问题主要还是通过路径压缩到祖先节点而不是父节点来解决的。
判断合法性
对于直接连向 \(U\) 的连通块,最终的结果一定是 \(U\)。
其余如果出现矛盾的连通块,最终的结果也一定全部是 \(U\)。这个的实现,只需要每次合并的时候判断一下到祖先节点的距离就好了。
时间复杂度 \(O(tn)\)。
代码
#include <bits/stdc++.h>
#define fi first
#define se second
#define lc (p<<1)
#define rc ((p<<1)|1)
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pi;
int n,m,tof[100005],tod[100005],f[100005],d[100005];
//n+1: T ,n+2: U
bitset<100005>ilg;
int findf(int x)
{
if(f[x]!=x)
{
int orif=f[x];
f[x]=findf(f[x]);
d[x]=(d[orif]^d[x]);
}
return f[x];
}
void combine(int x,int y,int td)
{
int fx=findf(x),fy=findf(y);
if(fx!=fy)
{
d[fx]=(d[y]^td^d[x]);
f[fx]=fy;
}
}
void init()
{
for(int i=1;i<=n+2;i++)f[i]=i,d[i]=0;
}
void solve()
{
cin>>n>>m;
for(int i=1;i<=n+2;i++)
{
tof[i]=i;
tod[i]=0;
}
ilg.reset();
for(int i=1;i<=m;i++)
{
int a,b;
char v;
cin>>v>>a;
if(v=='+')
{
cin>>b;
tof[a]=tof[b];
tod[a]=tod[b];
}
else if(v=='-')
{
cin>>b;
tof[a]=tof[b];
tod[a]=(tod[b]^1);
}
else if(v=='T')
{
tof[a]=n+1;
tod[a]=0;
}
else if(v=='F')
{
tof[a]=n+1;
tod[a]=1;
}
else
{
tof[a]=n+2;
tod[a]=0;
}
}
init();
for(int i=1;i<=n;i++)
{
if(findf(i)==findf(tof[i]))
{
if(tod[i]!=(d[i]^d[tof[i]]))ilg[findf(i)]=1;
}
combine(i,tof[i],tod[i]);
}
int ans=0;
for(int i=1;i<=n;i++)if(findf(i)==n+2||ilg[findf(i)])ans++;
cout<<ans<<'\n';
}
int main()
{
//freopen("tribool.in","r",stdin);
//freopen("tribool.out","w",stdout);
ios::sync_with_stdio(0);
cin.tie(0);
cout.tie(0);
int x,t;
cin>>x>>t;
while(t--)solve();
return 0;
}
Luogu P9869 NOIp2023 三值逻辑 题解 [ 绿 ] [ 带权并查集 ]的更多相关文章
- Codeforces Round #181 (Div. 2) B. Coach 带权并查集
B. Coach 题目连接: http://www.codeforces.com/contest/300/problem/A Description A programming coach has n ...
- bzoj3376/poj1988[Usaco2004 Open]Cube Stacking 方块游戏 — 带权并查集
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3376 题目大意: 编号为1到n的n(1≤n≤30000)个方块正放在地上.每个构成一个立方 ...
- Bzoj1202/洛谷P2294 [HNOI2005]狡猾的商人(带权并查集/差分约束系统)
题面 Bzoj 洛谷 题解 考虑带权并查集,设\(f[i]\)表示\(i\)的父亲(\(\forall f[i]<i\)),\(sum[i]\)表示\(\sum\limits_{j=fa[i]} ...
- 【bzoj3376-方块游戏】带权并查集
题意: n块积木,m个操作或询问.每次移动积木的时候,约翰会选择两块积木X,Y,把X搬到Y的上方.如果X已经和其它积木叠在一起了,那么应将这叠积木整体移动到Y的上方:如果Y已经和其它积木叠在一起了的, ...
- 并查集——poj2236(带权并查集)
题目:Wireless Network 题意:给定n台已损坏计算机的位置和计算机最远通信距离d,然后分别根据命令执行以下两种操作: "O p" (1 <= p <= N ...
- A Bug's Life POJ - 2492 (带权并查集)
A Bug's Life POJ - 2492 Background Professor Hopper is researching the sexual behavior of a rare spe ...
- [poj 2912] Rochambeau 解题报告 (带权并查集)
题目链接:http://poj.org/problem?id=2912 题目: 题目大意: n个人进行m轮剪刀石头布游戏(0<n<=500,0<=m<=2000) 接下来m行形 ...
- luogu 1196 银河英雄传说 带权并查集
带权并查集,其实有点像许多队列问情况的小学奥数 #include<bits/stdc++.h> #define rep(i,x,y) for(register int i=x;i<= ...
- luogu 2294 狡猾的商人 带权并查集
此题做法多啊 带权并查集,区间dp,前缀和,差分约束 1.自己写的前缀和, 11 #include<bits/stdc++.h> #define rep(i,x,y) for(regist ...
- NOI2002银河英雄传说-带权并查集
[NOI2002]银河英雄传说-带权并查集 luogu P1196 题目描述 Description: 公元五八○一年,地球居民迁至金牛座α第二行星,在那里发表银河联邦创立宣言,同年改元为宇宙历元年, ...
随机推荐
- Redis迁移工具之Redis-shake
Redis-shake is a tool for synchronizing data between two redis databases. Redis-shake是一个用于在两个redis之间 ...
- pve 下的群晖虚拟机硬盘空间扩容的记录
pve下,105号群晖虚拟机,btrfs系统,sata硬盘. 虚拟机容量硬盘130G,扩展至140G,还需要命令行和网页存储管理器界面操作,以实现扩容的目的. df -h Filesystem Siz ...
- mongo设置可选的唯一索引
对于一些可能为空但必须唯一的字段,mongo支持创建部分索引,使用 partialFilterExpression 字段. MongoDB version >= 3.2 下面以 unique_a ...
- LR语法分析算法
LR语法分析器 组成:一个输入,一个输出,状态栈,驱动程序,语法分析表 注意:规约后需要寻找新的符号在栈顶状态上的转换 例如: 状态栈 符号栈 输入 0 5 $id ...
- vscode实现Markdown实时预览
vscode - 插件搜索: Markdown Preview Enhanced 安装 然后打开vscode预览框,即可. That's ALL
- node 生成api文档 使用apidoc
先给大家看一下使用apidoc生成文档之后的视图,如果是所需的样子,可以继续往下看, 开始操作 1. 需要全局安装apidoc npm i apidoc -g 2. 生成配置文件 两种方法(用哪个都可 ...
- win10 ocx控件注册失败的解决办法
首先注意:ocx放在大部分系统目录无法注册,比如program file ,但windows目录可以. 第一步:关闭防火墙 第二步:确定是否缺少控件运行需要的微软运行库或其他运行环境 这里提供一个挺全 ...
- 【单片机】初次实验:Keil51的使用
哔哩哔哩/CSDN/博客园:萌狼蓝天 延时器 delay(int count){ int i,j; for(i=0;i<count;i++){ for(j=0;j<1000;j++); } ...
- Gitlab 实现仓库完全迁移
方法一:最快 gitlab用url导入注意事项看图 方法二 首先需要在新的服务服务器上新建一个项目 然后用 Git Bash 执行以下命令 git clone --mirror 项目原代码仓库地址 / ...
- Qt/C++音视频开发59-使用mdk-sdk组件/原qtav作者力作/性能凶残/超级跨平台
一.前言 最近一个月一直在研究mdk-sdk音视频组件,这个组件是原qtav作者的最新力作,提供了各种各样的示例demo,不仅限于支持C++,其他各种比如java/flutter/web/androi ...