P4332 [SHOI2014]三叉神经树
\(\color{#0066ff}{ 题目描述 }\)
计算神经学作为新兴的交叉学科近些年来一直是学术界的热点。一种叫做SHOI 的神经组织因为其和近日发现的化合物 SHTSC 的密切联系引起了人们的极大关注。
SHOI 组织由若干个 SHOI 细胞构成,SHOI 细胞之间形成严密的树形结构。每个 SHOI 细胞都有且只有一个输出端,被称为轴突,除了一个特殊的、被称为根细胞的 SHOI 细胞的输出作为整个组织的输出以外,其余细胞的轴突均连向其上级 SHOI 细胞;并且有且只有三个接收端,被称为树突,从其下级细胞或者其它神经组织那里接收信息。SHOI 细胞的信号机制较为简单,仅有 \(0\)和 \(1\)两种。每个 SHOI 细胞根据三个输入端中 \(0\)和 \(1\)信号的多寡输出较多的那一种。
现在给出了一段 SHOI 组织的信息,以及外部神经组织的输入变化情况。请你模拟 SHOI 组织的输出结果。
\(\color{#0066ff}{输入格式}\)
输入的第一行包含一个整数 \(n\)。表示 SHOI 组织的总细胞个数。SHOI 细胞由 \(1\)~ \(n\) 编号,编号为 \(1\) 的是根细胞。
从第二行开始的 \(n\)行,每行三个整数 \(x_1, x_2, x_3\),分别表示编号为 \(1\)~ \(n\) 的 SHOI 细胞的树突连接。\(1 < x_i \leq n\) 表示连向编号为 \(x_i\) 的细胞的轴突, \(n < x_i \leq 3n+1\) 表示连向编号为 \(x_i\) 的外界输入。输入数据保证给出的 SHOI 组织是合法的,且所有的 \(x_i\) 两两不同。
接下来一行包含 \(2n+1\) 个 \(0/1\) 的整数,表示初始时的外界输入。
第 \(n+3\)行有一个整数 \(q\),表示总操作数。
之后 \(q\) 行每行一个整数 \(x\),表示编号为 \(x\) 的外界输入发生了变化。
\(\color{#0066ff}{输出格式}\)
输出 \(q\) 行,每行一个整数,对应第 \(i\) 次外界输入变化后的根细胞的输出。
\(\color{#0066ff}{输入样例}\)
3
2 3 4
5 6 7
8 9 10
0 0 0 0 1 1 1
5
4
4
5
6
8
\(\color{#0066ff}{输出样例}\)
1
0
0
1
1
\(\color{#0066ff}{数据范围与提示}\)
对于 \(10 \%\)的数据, \(n \leq 1000, q \leq 1000\)。
对于 \(30 \%\)的数据, \(n \leq 100000, q \leq 100000\)。
对于 \(100 \%\)的数据, \(n \leq 500000, q \leq 500000\)。
\(\color{#0066ff}{ 题解 }\)
这毒瘤题读了半天没看懂啥意思,现在简单概括一下题意
给你一颗三叉树(类比二叉树qwq),1为根,前n个点每个点有三个孩子,\(n+1\to3n\)这些点为叶子节点,然后给你所有叶子的权值0/1,定义\(1-n\)点的权值为,三个孩子中权为1的孩子的个数是否大于1,如果是则权值为1,否则为0。现在给你一些操作,每次让一个叶子的权值异或1,每次操作输出根的权值
我们考虑每次操作对答案的影响,设\(v_i\)为i有多少个权值为1的孩子\(v_i\in [0,3]\)
如果当前点权值由0变1,那么什么情况下父亲的权值也会改变呢
显然当且仅当父亲的v=1,那么加上自己的改变, 父亲的v变为2,这时父亲的权值也从原来的0变为1,这就是一个子问题了
同理,如果是1变0, 当且仅当父亲的v=2时父亲也会改变
那么,我们可以得到一个结论,每次修改一个点产生的影响是从这个点向上一条连续v为1或2的链的权值的改变,只要这个链包括了1, 答案就会改变
那么现在的问题是,怎么快速维护这个东西,我们需要快速找到一个点向上延伸1/2的最上面那个点,还有一条链的权值的修改
可以用LCT维护这个东西, 显然不需要makeroot操作,那么翻转标记就没用了
每个点维护两个点的指针,一个是最深的v!=1的点,一个是最深的v!=2的点
这个点实际上就是延伸的链顶的父亲
为什么要最深? 因为要保证下面的1/2连续
怎么维护? 因为要最深,所以在splay上,先考虑右儿子,然后是自己,最后是左儿子
如果这个点存在,那么就对答案没有影响(仔细想想),我们把这个点转上去,那么实际上就是当前点的单点修改,右子树整体的修改,这个打标记即可,注意v改变了,n1和n2要互换!
如果不存在呢,说明此链一直延伸到根,那么直接对整个splay打标记,并把答案修改即可
这题就没了。。。
发现实际上每个点的权值就是\(\frac v 2\),所以直接维护v就行了
#include<bits/stdc++.h>
#define LL long long
LL in() {
char ch; LL x = 0, f = 1;
while(!isdigit(ch = getchar()))(ch == '-') && (f = -f);
for(x = ch ^ 48; isdigit(ch = getchar()); x = (x << 1) + (x << 3) + (ch ^ 48));
return x * f;
}
const int maxn = 5e5 + 3;
const int maxm = 1.5e6 + 3;
struct node {
node *ch[2], *fa, *n1, *n2;
int val, tag;
node(int val = 0, int tag = 0): val(val), tag(tag) {
ch[0] = ch[1] = fa = n1 = n2 = NULL;
}
void trn() { std::swap(n1, n2); val ^= 3; tag ^= 1; }
void dwn() {
if(!tag) return;
if(ch[0]) ch[0]->trn();
if(ch[1]) ch[1]->trn();
tag = 0;
}
void upd() {
n1 = n2 = NULL; //这行不能少!
//右根左这样找
if(ch[1]) n1 = ch[1]->n1;
if(!n1) n1 = this->val == 1? NULL : this;
if(ch[0] && !n1) n1 = ch[0]->n1;
if(ch[1]) n2 = ch[1]->n2;
if(!n2) n2 = this->val == 2? NULL : this;
if(ch[0] && !n2) n2 = ch[0]->n2;
}
bool isr() { return this == fa->ch[1]; }
bool ntr() { return fa && (fa->ch[1] == this || fa->ch[0] == this); }
}pool[maxn];
int fa[maxm], val[maxm], du[maxm];
int n, ans;
void toposort() {
std::queue<int> q;
for(int i = n + 1; i <= 3 * n + 1; i++) val[i] = in() << 1, q.push(i);
while(!q.empty()) {
int tp = q.front(); q.pop();
if(tp <= n) pool[tp].upd(), pool[tp].fa = pool + fa[tp];
pool[fa[tp]].val += (tp > n? val[tp] : pool[tp].val) >> 1;
du[fa[tp]]--;
if(!du[fa[tp]]) q.push(fa[tp]);
}
pool[1].fa = NULL;
}
void rot(node *x) {
node *y = x->fa, *z = y->fa;
bool k = x->isr(); node *w = x->ch[!k];
if(y->ntr()) z->ch[y->isr()] = x;
(x->ch[!k] = y)->ch[k] = w;
(y->fa = x)->fa = z;
if(w) w->fa = y;
y->upd(), x->upd();
}
void splay(node *o) {
static node *st[maxn];
int top;
st[top = 1] = o;
while(st[top]->ntr()) st[top + 1] = st[top]->fa, top++;
while(top) st[top--]->dwn();
while(o->ntr()) {
if(o->fa->ntr()) rot(o->isr() ^ o->fa->isr()? o : o->fa);
rot(o);
}
}
void access(node *x) {
for(node *y = NULL; x; x = (y = x)->fa)
splay(x), x->ch[1] = y, x->upd();
}
int main() {
n = in();
for(int i = 1; i <= n; i++) du[fa[in()] = fa[in()] = fa[in()] = i] = 3;
toposort();//求初始的答案
ans = pool[1].val >> 1;
int x, tp;
for(int T = in(); T --> 0;) {
tp = (val[x = in()] ^= 2) >> 1;
access(pool + (x = fa[x])), splay(pool + x);
node *o = pool + x;
node *nr = tp? o->n1 : o->n2; //找最深的点
if(nr) {
splay(nr);
if(nr->ch[1]) nr->ch[1]->trn(), nr->ch[1]->upd();
nr->val += tp? 1 : -1;
nr->upd();
}
else ans ^= 1, o->trn(), o->upd();
printf("%d\n", ans);
}
return 0;
}
P4332 [SHOI2014]三叉神经树的更多相关文章
- 洛谷P4332 [SHOI2014]三叉神经树(LCT,树剖,二分查找,拓扑排序)
洛谷题目传送门 你谷无题解于是来补一发 随便百度题解,发现了不少诸如树剖\(log^3\)LCT\(log^2\)的可怕描述...... 于是来想想怎么利用题目的性质,把复杂度降下来. 首先,每个点的 ...
- P4332 [SHOI2014]三叉神经树(LCT)
Luogu4332 LOJ2187 题解 代码-Tea 题意 : 每个点有三个儿子 , 给定叶节点的权值\(0\)或\(1\)且支持修改 , 非叶子节点的权值为当有\(>=2\)个儿子的权值为\ ...
- 洛谷P4332 [SHOI2014]三叉神经树(LCT)
传送门 FlashHu大佬太强啦%%% 首先,我们可以根据每一个点的权值为$1$的儿子的个数把每个点记为$0~3$,表示这一个点的点权 先考虑一下暴力的过程,假设从$0$变为$1$,先更改一个叶子结点 ...
- [BZOJ 3553][SHOI2014]三叉神经树
传送门(下面也有题面) 题目大意: 一颗有根树,每个非叶子节点都有三个子节点,每个节点的权为0/1. 每个节点的权 取决于其所有子节点中 哪种权出现的次数更多. 有若干次询问,每次询问修改一个叶子节点 ...
- BZOJ3553 : [Shoi2014]三叉神经树
设val[i]为i连出去的树突中输出值为0的个数 如果val[x]<=1,输出值为1,否则输出值为0 修改x就相当于val[f[i]]++或者val[f[i]]-- 用Link-cut Tree ...
- 3553: [Shoi2014]三叉神经树(树链剖分)
这道题特别恶心,首先我们可以发现更改的就是出现连续的一或二,那么就用线段树+树链剖分找到这个范围 想到是不难想,就是打起来恶心罢了= = CODE: #include<cstdio> #i ...
- [SHOI2014]三叉神经树
题目描述 计算神经学作为新兴的交叉学科近些年来一直是学术界的热点.一种叫做SHOI 的神经组织因为其和近日发现的化合物 SHTSC 的密切联系引起了人们的极大关注. SHOI 组织由若干个 SHOI ...
- BZOJ 3553: [Shoi2014]三叉神经树 LCT
犯傻了,想到了如果是 0->1 的话就找最深的非 1 编号,是 1 -> 0 的话就找最深的非 0 编号. 但是没有想到这个东西可以直接维护. 假设不考虑叶子节点,那么如果当前点的值是 1 ...
- 洛谷4322 SHOI2014 三叉神经树(LCT+思维)
好久之前做的题了QWQ 现在来补一发博客 一道神仙题啊..qwq 首先,我们可以看出来,我们如果对于每个点维护一个\(val\),表示他的直系儿子中有几个表现为1的. 那么\(val[x]>&g ...
随机推荐
- maven中maven dependencies中依赖出现了项目
maven 中maven dependencies中依赖出现了项目,把依赖的项目关掉,项目消失,但是还是无法打包 ,出现的错误如图.说明:依赖的项目为project-dao 打包的项目为projec ...
- Appium—python_ 安卓手机划屏幕操作
开始的时候 不知道 python_unittest框架的命名规则,导致方法进不去,后来 改变方法名 能获取 # conding=utf- from appium import webdriver im ...
- 类型:Java;问题:eclipse配置maven;结果:eclipse配置maven
eclipse配置maven 下面跟大家分享的是eclipse配置maven的方法. 方法/步骤 安装maven之前,要先安装jdk及配置JAVA_HOME环境变量.JDK1.4以上. 下载maven ...
- C语言学习笔记--#error 、 #line 和 #pragma 的使用
1. #error 的用法 (1)#error 是一种预编译器指示字,用于生成一个编译错误消息 (2)用法:#error message //注意:message 不需要用双引号包围 (3)#erro ...
- object类型对象 ref参数如何理解?
class Program { static void Main(string[] args) { Student stu = new Student { Name = "老王" ...
- windows安装和配置JDK
安装完JDK后配置环境变量 计算机→属性→高级系统设置→高级→环境变量 系统变量→新建 JAVA_HOME 变量 . 变量值填写jdk的安装目录(本人是 E:\Java\jdk1.7.0) ...
- Tornado之抽屉实战(1)--分析与架构
项目模拟地址:http://dig.chouti.com/ 知识点应用: AJAX 用于偷偷发请求 原生ajax jQuery ajax($.ajax) iframe伪造 上传文件 传统Form ...
- ava的打包jar、war、ear包的作用、区别、打包方式
编为大家介绍,基于Java的打包jar.war.ear包的作用与区别详解.需要的朋友参考下以最终客户的角度来看,JAR文件就是一种封装,他们不需要知道jar文件中有多少个.class文件,每个文件中的 ...
- VS2013中,将Qt的GUI程序改为控制台程序
在Visual studio 中创建QT GUI程序是不带Console的,但是调试时候常常需要查看打印信息,可以通过如下设置显示控制台 方法一.在vs中直接创建控制台程序方法二.当你通过设置你的应用 ...
- Boost log中的几个问题
1. 使用动态库时,要定义 BOOST_LOG_DYN_LINK 或者 BOOST_ALL_DYN_LINK 否则会出现如下错误: CMakeFiles/xxxx.dir/xxxx.cpp.o: I ...