Max Mex
无法直接处理
可以二分答案!
[0,mid]是否在同一个链上?
可以不修改地做了
修改?
能不能信息合并?可以!
记录包含[l,r]的最短链的两端
可以[0,k][k+1,mid]合并:枚举四个端点中的两个,使得另外两个一定在这两个的路径上
(判断z点在x,y路径上:(lca(x,z)==z||lca(y,z)=z)&&(lca(lca(x,y),z)=lca(x,y))画图即可理解
能合并,所以线段树可以维护。
线段树维护
线段树上二分。
LCA用ST表存
#include<bits/stdc++.h>
#define reg register int
#define il inline
#define fi first
#define se second
#define mk(a,b) make_pair(a,b)
#define numb (ch^'0')
using namespace std;
typedef long long ll;
template<class T>il void rd(T &x){
char ch;x=;bool fl=false;
while(!isdigit(ch=getchar()))(ch=='-')&&(fl=true);
for(x=numb;isdigit(ch=getchar());x=x*+numb);
(fl==true)&&(x=-x);
}
template<class T>il void output(T x){if(x/)output(x/);putchar(x%+'');}
template<class T>il void ot(T x){if(x<) putchar('-'),x=-x;output(x);putchar(' ');}
template<class T>il void prt(T a[],int st,int nd){for(reg i=st;i<=nd;++i) ot(a[i]);putchar('\n');} namespace Miracle{
const int N=2e5+;
int n,q;
struct node{
int nxt,to;
}e[*N];
int hd[N],cnt;
void add(int x,int y){
e[++cnt].nxt=hd[x];
e[cnt].to=y;
hd[x]=cnt;
}
int dep[N];
int a[*N],tot;
int dfn[N];
int f[*N][];
int lg[*N];
int id[N],fid[N];
void dfs(int x,int d){
dep[x]=d;
a[++tot]=x;
dfn[x]=tot;
for(reg i=hd[x];i;i=e[i].nxt){
int y=e[i].to;
dfs(y,d+);
a[++tot]=x;
}
}
int big(int x,int y){
return dep[x]<dep[y]?x:y;
}
int lca(int x,int y){
if(dfn[x]>dfn[y]) swap(x,y);
int len=lg[dfn[y]-dfn[x]+];
return big(f[dfn[x]][len],f[dfn[y]-(<<len)+][len]);
}
int on(int p1,int p2,int p3){
int y=lca(p1,p2);
// cout<<" p1 "<<p1<<" p2 "<<p2<<" p3 "<<p3<<" y "<<y<<endl;
return ((lca(p1,p3)==p3||lca(p2,p3)==p3)&&(lca(y,p3)==y));
}
struct tr{
int p[];
int can;
void init(){
can=p[]=p[]=;
}
void op(){
cout<<" can "<<can<<" p[0] "<<p[]<<" p[1] "<<p[]<<endl;
}
}t[*N];
#define ls (x<<1)
#define rs (x<<1|1)
#define mid ((l+r)>>1)
void pushup(int x){
t[x].init();
if(t[x<<].can&&t[x<<|].can){
t[x].can|=on(t[ls].p[],t[ls].p[],t[rs].p[])&&on(t[ls].p[],t[ls].p[],t[rs].p[]);
if(t[x].can==) {t[x].p[]=t[ls].p[],t[x].p[]=t[ls].p[];return;}
t[x].can|=on(t[rs].p[],t[rs].p[],t[ls].p[])&&on(t[rs].p[],t[rs].p[],t[ls].p[]);
if(t[x].can==) {t[x].p[]=t[rs].p[],t[x].p[]=t[rs].p[];return;}
if(!t[x].can){
for(reg i=;i<=;++i){
for(reg j=;j<=;++j){
t[x].can|=on(t[ls].p[i],t[rs].p[j],t[ls].p[i^])&&on(t[ls].p[i],t[rs].p[j],t[rs].p[j^]);
if(t[x].can==) {t[x].p[]=t[ls].p[i],t[x].p[]=t[rs].p[j];return;}
}
}
}
}
}
tr merge(tr A,tr B){
tr C;C.can=C.p[]=C.p[]=;
// cout<<" A ";A.op();
// cout<<" B ";B.op();
if(A.can&&B.can){
C.can|=on(A.p[],A.p[],B.p[])&&on(A.p[],A.p[],B.p[]);
if(C.can==) {C.p[]=A.p[],C.p[]=A.p[];return C;}
C.can|=on(B.p[],B.p[],A.p[])&&on(B.p[],B.p[],A.p[]);
if(C.can==) {C.p[]=B.p[],C.p[]=B.p[];return C;}
if(!C.can){
for(reg i=;i<=;++i){
for(reg j=;j<=;++j){
C.can|=on(A.p[i],B.p[j],A.p[i^])&&on(A.p[i],B.p[j],B.p[j^]);
if(C.can==) {C.p[]=A.p[i],C.p[]=B.p[j];return C;}
}
}
}
}
return C;
}
void build(int x,int l,int r){
if(l==r){
t[x].p[]=t[x].p[]=fid[l];
t[x].can=;return;
}
t[x].can=;
build(x<<,l,mid);
build(x<<|,mid+,r);
pushup(x);
}
void upda(int x,int l,int r,int p,int c){//pos to c
if(l==r){
t[x].p[]=c;t[x].p[]=c;
t[x].can=;return;
}
if(p<=mid) upda(ls,l,mid,p,c);
else upda(rs,mid+,r,p,c);
pushup(x);
}
tr tmp;
int query(int x,int l,int r){
// cout<<" query "<<x<<" "<<l<<" "<<r<<" : ";t[x].op(); tr C;
if(l==r){
if(tmp.can==-){
C=t[x];
}else{
C=merge(tmp,t[x]);
}
return C.can==?l:l-;
}
// cout<<" ls ";t[ls].op();
if(tmp.can==-){
C=t[ls];
}else{
C=merge(tmp,t[ls]);
}
// C.op();
if(C.can){
tmp=C;
return query(rs,mid+,r);
}else{
return query(ls,l,mid);
}
}
int main(){
rd(n);
for(reg i=;i<=n;++i){
rd(id[i]);fid[id[i]]=i;
}
int y;
for(reg i=;i<=n;++i){
rd(y);add(y,i);
}
dfs(,);
for(reg i=;i<=*n-;++i){
lg[i]=(i>>(lg[i-]+))?lg[i-]+:lg[i-];
f[i][]=a[i];
}
// cout<<" tot "<<tot<<endl;
// prt(a,1,tot);
// prt(lg,1,2*n-1);
for(reg j=;j<=;++j){
for(reg i=;i+(<<j)-<=tot;++i){
f[i][j]=big(f[i][j-],f[i+(<<(j-))][j-]);
}
}
build(,,n-); rd(q);
int op,x;
while(q--){
rd(op);
if(op==){
rd(x);rd(y);
upda(,,n-,id[x],y);
upda(,,n-,id[y],x);
swap(id[x],id[y]);
}else{
tmp.init();
tmp.can=-;
printf("%d\n",query(,,n-)+);
}
}
return ;
} }
signed main(){
Miracle::main();
return ;
} /*
Author: *Miracle*
Date: 2019/4/11 16:56:19
*/
正难则反考虑二分检验。
和[IOI2018] seats 排座位有点像,维护前缀信息的连通性
修改?能否两两区间合并?树上链的信息合并的套路。
Max Mex的更多相关文章
- Codeforces 1083C Max Mex
Description 一棵\(N\)个节点的树, 每个节点上都有 互不相同的 \([0, ~N-1]\) 的数. 定义一条路径上的数的集合为 \(S\), 求一条路径使得 \(Mex(S)\) 最大 ...
- CF1083C Max Mex 线段树
题面 CF1083C Max Mex 题解 首先我们考虑,如果一个数x是某条路径上的mex,那么这个数要满足什么条件? 1 ~ x - 1的数都必须出现过. x必须没出现过. 现在我们要最大化x,那么 ...
- CF 1083 C. Max Mex
C. Max Mex https://codeforces.com/contest/1083/problem/C 题意: 一棵$n$个点的树,每个点上有一个数(每个点的上的数互不相同,而且构成一个0~ ...
- CodeForces 1084 F Max Mex
Max Mex 题意:问在树上的所有路中mex值最大是多少. 题解: 用线段树维护值. 区间[L,R]意味着 区间[L,R]的数可不可以合并. 重点就是合并的问题了. 首先合法的区间只有3种: 1. ...
- CF 526F Max Mex(倍增求LCA+线段树路径合并)
Max Mex 题目地址:https://codeforces.com/contest/1084/problem/F 然后合并时注意分情况讨论: 参考代码: #include<bits/stdc ...
- 【Codeforces 1083C】Max Mex(线段树 & LCA)
Description 给定一颗 \(n\) 个顶点的树,顶点 \(i\) 有点权 \(p_i\).其中 \(p_1,p_2,\cdots, p_n\) 为一个 \(0\sim (n-1)\) 的一个 ...
- [CF1083C]Max Mex
题目大意:有一棵$n(n\leqslant2\times10^5)$个点的树,每个点有点权,所有的点权构成了$0\sim n-1$的排列.$q(q\leqslant2\times10^5)$次操作,操 ...
- 【线段树】【CF1083C】 Max Mex
Description 给定一棵有 \(n\) 个点的树,每个节点有点权.所有的点权构成了一个 \(0~\sim~n - 1\) 的排列.有 \(q\) 次操作,每次操作 \(1\) 为交换两个点的点 ...
- Codeforces 1083C Max Mex [线段树]
洛谷 Codeforces 思路 很容易发现答案满足单调性,可以二分答案. 接下来询问就转换成判断前缀点集是否能组成一条链. 我最初的想法:找到点集的直径,判断直径是否覆盖了所有点,需要用到树套树,复 ...
随机推荐
- Mybatis Dao层注解及XML组合Dao的开发方式
mybatis可以用xml进行数据操作,也可以在dao层用注解的方式,也可以采取xml和dao层接口组合使用的方法.显然 ,后者更加简单. 实体类Student package com.zhao. ...
- 初次启动hive,解决 ls: cannot access /home/hadoop/spark-2.2.0-bin-hadoop2.6/lib/spark-assembly-*.jar: No such file or directory问题
>>提君博客原创 http://www.cnblogs.com/tijun/ << 刚刚安装好hive,进行第一次启动 提君博客原创 [hadoop@ltt1 bin]$ ...
- C#读书笔记:线程,任务和同步
前言 学习C#两个多月了,像当初实习做PHP开发一样,也是由着一个个Feature需求,慢慢掌握了很多相关的编程技巧.本次主要记录下学习C# 多线程的相关知识. 参考书籍:<Csharp高级编程 ...
- spring boot session error
Error starting ApplicationContext. To display the conditions report re-run your application with 'de ...
- 莫烦keras学习自修第二天【backend配置】
keras的backend包括tensorflow和theano,tensorflow只能在macos和linux上运行,theano可以在windows,macos及linux上运行 1. 使用配置 ...
- DataSet 取值,DataSet行数,DataSet列数 从DataSet中取出特定值
1 DataSet.Table[0].Rows[ i ][ j ] 其中i 代表第 i 行数, j 代表第 j 列数 2 DataSet.Table[0].Rows[ i ].ItemArray[ j ...
- dataTable之自定义按钮实现全表 复制 打印 导出 重载
//本文对常用表格插件datatable 的自定义按钮功能键进行详细解释//其中 15-78行是定义表单//16 18 19 三行定义自定义功能按钮 实现对全表的 复制 打印 导出(csv即excel ...
- Ubuntu16.04 启动纯文本界面方法
问题: Ubuntu16.04 如何启动纯文本界面. 解决方法: 1.系统启动后,在登陆界面点击Ctrl+Shift+F1切换到文本登陆界面: 2.修改为默认从文本界面登陆: sudo vi /etc ...
- hdu-4300(kmp或者拓展kmp)
题意:乱七八糟说了一大堆,就是先给你一个长度26的字符串,对应了abcd....xyz,这是一个密码表.然后给你一个字符串,这个字符串是不完整的(完整的应该是前半部分是加密的,后半部分是解密了的),然 ...
- [SimplePlayer] 实现一个简单的播放器
简单的播放器需要实现一个最基本的功能:播放视频文件. 实现这个功能需要包含以下几个步骤: 从视频文件中提取视频图像 在屏幕上显示视频图像 视频帧的同步,也就是保证视频图像在合适的时间在屏幕上显示 从视 ...