【LOJ6498】「雅礼集训 2018 Day2」农民
solution
直接暴力模拟,原数据可获得满分的成绩。
对于每个点,其父亲对其都有一个限制。故我们只需要判断当前点到根的路径上的限制是否都能满足即可。
考虑用树剖+线段树维护这个限制。考虑到翻转操作,我们需维护当前区间左儿子限制最小值/最大值和右儿子最小值/最大值。
注意翻转操作会导致树的形态的改变。
code
#include<cstdio>
#include<algorithm>
using namespace std;
namespace io {
const int SIZE=(1<<21)+1;
char ibuf[SIZE],*iS,*iT,c;
#define gc()(iS==iT?(iT=(iS=ibuf)+fread(ibuf,1,SIZE,stdin),(iS==iT?EOF:*iS++)):*iS++)
inline int gi (){
int x=0,f=1;
for(c=gc();c<'0'||c>'9';c=gc())if(c=='-')f=-1;
for(;c<='9'&&c>='0';c=gc()) x=(x<<1)+(x<<3)+(c&15); return x*f;
}
} using io::gi;
const int N=1e5+5,inf=1<<30;
int n,m,a[N],ls[N],rs[N],size[N],son[N],top[N],dep[N],fa[N],id[N],ed[N],idx[N],tim,rt;
int mxl[N<<2],mnl[N<<2],mxr[N<<2],mnr[N<<2];
bool nr[N],isl[N],rev[N<<2],tg[N<<2];
#define lx (x<<1)
#define rx (x<<1|1)
void dfs(int u)
{
if(!u) return ;
size[u]=1;
fa[ls[u]]=u,fa[rs[u]]=u;
dep[ls[u]]=dep[rs[u]]=dep[u]+1;
dfs(ls[u]),dfs(rs[u]);
son[u]=(size[ls[u]]>size[rs[u]]?ls[u]:rs[u]);
}
void dfs2(int u, int tp)
{
if(!u) return ;
id[u]=++tim;
top[u]=tp, idx[tim]=u;
isl[ls[u]]=true;
if(son[u]==ls[u]) dfs2(ls[u],tp),dfs2(rs[u],rs[u]);
else dfs2(rs[u],tp),dfs2(ls[u],ls[u]);
ed[u]=tim;
}
void pushup(int x)
{
mxl[x]=max(mxl[lx],mxl[rx]);
mnl[x]=min(mnl[lx],mnl[rx]);
mxr[x]=max(mxr[lx],mxr[rx]);
mnr[x]=min(mnr[lx],mnr[rx]);
}
void pushdown(int x)
{
if(!rev[x]) return ;
rev[lx]^=1,rev[rx]^=1;
tg[lx]^=1,tg[rx]^=1;
swap(mxl[lx],mxr[lx]);
swap(mnl[lx],mnr[lx]);
swap(mxl[rx],mxr[rx]);
swap(mnl[rx],mnr[rx]);
rev[x]=0;
}
void upd(int x, int u, int w)
{
if(u==1) mxl[x]=mxr[x]=-1,mnl[x]=mnr[x]=inf;
else if(isl[u]^tg[x]) mxl[x]=mnl[x]=w,mxr[x]=-1,mnr[x]=inf;
else mxr[x]=mnr[x]=w,mxl[x]=-1,mnl[x]=inf;
}
void build(int x, int l, int r)
{
if(l==r)
{
upd(x,idx[l],a[fa[idx[l]]]);
return ;
}
int mid=l+r>>1;
build(lx,l,mid),build(rx,mid+1,r);
pushup(x);
}
void update(int x, int l, int r, int s, int w)
{
if(l==r)
{
upd(x,idx[l],w);
return ;
}
pushdown(x);
int mid=l+r>>1;
s<=mid?update(lx,l,mid,s,w):update(rx,mid+1,r,s,w);
pushup(x);
}
void rever(int x, int l, int r, int sl, int sr)
{
if(sl<=l&&r<=sr)
{
rev[x]^=1,tg[x]^=1;
swap(mxl[x],mxr[x]);
swap(mnl[x],mnr[x]);
return ;
}
pushdown(x);
int mid=l+r>>1;
if(sl<=mid) rever(lx,l,mid,sl,sr);
if(sr>mid) rever(rx,mid+1,r,sl,sr);
pushup(x);
}
pair<int,int> query(int x, int l, int r, int sl, int sr)
{
if(sl<=l&&r<=sr) return make_pair(mnl[x],mxr[x]);
pushdown(x);
int mid=l+r>>1;
if(sr<=mid) return query(lx,l,mid,sl,sr);
else if(sl>mid) return query(rx,mid+1,r,sl,sr);
else
{
pair<int,int> ql=query(lx,l,mid,sl,sr),qr=query(rx,mid+1,r,sl,sr);
int ans_mnl,ans_mxr;
ans_mnl=min(ql.first,qr.first);
ans_mxr=max(ql.second,qr.second);
return make_pair(ans_mnl,ans_mxr);
}
}
bool check(int x)
{
int ans_mnl=inf,ans_mxr=-1,w=a[x];
while(top[x]^top[rt])
{
pair<int,int> q=query(1,1,n,id[top[x]],id[x]);
ans_mnl=min(ans_mnl,q.first);
ans_mxr=max(ans_mxr,q.second);
x=fa[top[x]];
}
pair<int,int> q=query(1,1,n,id[rt],id[x]);
ans_mnl=min(ans_mnl,q.first);
ans_mxr=max(ans_mxr,q.second);
return (w<ans_mnl&&w>ans_mxr);
}
int main()
{
n=gi(),m=gi();
for(int i=1;i<=n;++i) a[i]=gi(),ls[i]=gi(),rs[i]=gi(),nr[ls[i]]=nr[rs[i]]=true;
for(int i=1;i<=n;++i) if(!nr[i]) rt=i;
dfs(rt),dfs2(rt,rt),build(1,1,n);
while(m--)
{
int op=gi(),x=gi();
if(op==1)
{
int w=gi();
if(ls[x]) update(1,1,n,id[ls[x]],w);
if(rs[x]) update(1,1,n,id[rs[x]],w);
a[x]=w;
}
if(op==2) rever(1,1,n,id[x]+1,ed[x]);
if(op==3) puts(check(x)?"YES":"NO");
}
}
【LOJ6498】「雅礼集训 2018 Day2」农民的更多相关文章
- 「雅礼集训 2018 Day2」农民
传送门 Description 「搞 OI 不如种田.」 小 D 在家种了一棵二叉树,第 ii 个结点的权值为 \(a_i\). 小 D 为自己种的树买了肥料,每天给树施肥. 可是几天后,小 D 却 ...
- LOJ6500. 「雅礼集训 2018 Day2」操作(哈希+差分)
题目链接 https://loj.ac/problem/6500 题解 区间取反 \(01\) 串的经典套路是差分.我们令 \(b_i = a_i\ {\rm xor}\ a_{i - 1}\)(\( ...
- 【卡常 bitset 分块】loj#6499. 「雅礼集训 2018 Day2」颜色
好不容易算着块大小,裸的分块才能过随机极限数据:然而这题在线的数据都竟然是构造的…… 题目描述 有 $n$ 个数字,第 $i$ 个数字为 $a_i$. 有 $m$ 次询问,每次给出 $k_i$ 个区间 ...
- #6499. 「雅礼集训 2018 Day2」颜色 [分块,倍增,bitset]
bitset压位,因为是颜色数,直接倍增,重合部分不管,没了. // powered by c++11 // by Isaunoya #include <bits/stdc++.h> #d ...
- 「LOJ #6500」「雅礼集训 2018 Day2」操作
description LOJ 6500 solution 根据常有套路,容易想到将区间差分转化为异或数组上的单点修改,即令\(b_i=a_i \ xor\ a_{i-1}\), 那么将\([l,l+ ...
- 「雅礼集训 2017 Day2」解题报告
「雅礼集训 2017 Day2」水箱 我怎么知道这种题目都能构造树形结构. 根据高度构造一棵树,在树上倍增找到最大的小于约束条件高度的隔板,开一个 \(vector\) 记录一下,然后对于每个 \(v ...
- #6034. 「雅礼集训 2017 Day2」线段游戏 李超树
#6034. 「雅礼集训 2017 Day2」线段游戏 内存限制:256 MiB时间限制:1000 ms标准输入输出 题目类型:传统评测方式:Special Judge 上传者: 匿名 提交提交记录统 ...
- 【loj6034】「雅礼集训 2017 Day2」线段游戏
#6034. 「雅礼集训 2017 Day2」线段游戏 内存限制:256 MiB 时间限制:1000 ms 标准输入输出 题目类型:传统 评测方式:Special Judge 上传者: 匿名 题目描述 ...
- Loj #6503. 「雅礼集训 2018 Day4」Magic
Loj #6503. 「雅礼集训 2018 Day4」Magic 题目描述 前进!前进!不择手段地前进!--托马斯 · 维德 魔法纪元元年. 1453 年 5 月 3 日 16 时,高维碎片接触地球. ...
随机推荐
- leetcode刷题-- 4. 贪心
贪心 455分发饼干 假设你是一位很棒的家长,想要给你的孩子们一些小饼干.但是,每个孩子最多只能给一块饼干.对每个孩子 i ,都有一个胃口值 gi ,这是能让孩子们满足胃口的饼干的最小尺寸:并且每块饼 ...
- Java记录4--string
1.toString所有的类都默认自动继承了Objiect类 2.Object类中的toString方法返回的时类的名字和该哈希表码组成的一个字符串, System.out.println(类对象名) ...
- git 常用命令记录 -- 快捷&备忘
1.安装 略2.git拉取远程分支 git config user.name git config user.email git config --global user.name xxxx git ...
- Codeforces 1304B. Longest Palindrome
根据数据范围,暴力可以解决,对每一个串,找与其互为回文的串,或者判断自身是否为回文串,然后两两将互为回文的串排列在头尾,中间放且只能最多放一个自身为回文串的串,因为题目说每个串都是不同的 #inclu ...
- 一 注册功能&登录功能,权限拦截
注册功能: 前端JSP:提供表单注册信息以及访问路径,发送请求到Strus2. Struts2 : 通过模型驱动接收并封装User对象,Spring依赖注入(无参构造+setter方法)获取业务层Us ...
- A. Hotelier
A. Hotelier 题意:一家有10间房间的旅馆(10个房间排成一排),在旅馆的左右两端都有一个办理入住的前台,L代表在左端办理入住,R代表在右端办理入住,顾客都会挑选距离最近的空房间入住,数 ...
- markdown基本语法教程
标题 一级标题 二级标题 三级标题 以此类推,总共六级标题,建议在警号后面加一个空格,这是最标准的markdown语法 列表 在markdown下: 列表的显示只需要在文字前加上-.+或*即可变为无序 ...
- Linux-使用之vim编译安装出现的问题
---恢复内容开始--- cd <directory> Short for "change directory". The shorthand name for the ...
- 移动互联网APP测试流程及测试点
1.2测试周期 测试周期可按项目的开发周期来确定测试时间,一般测试时间为两三周(即15个工作日),根据项目情况以及版本质量可适当缩短或延长测试时间.正式测试前先向主管确认项目排期. 1.3测试资源 测 ...
- redis学习笔记-01:redis简介
1.redis是一个高性能的Nosql数据库,遵守BSD协议,使用c语言编写.支持网络.可基于内存亦可持久化,是一种日志型.Key-Value数据库,也可看做是一个分布式的.基于内存的缓存工具. 2. ...