CF1083C Max Mex 线段树
题面
题解
首先我们考虑,如果一个数x是某条路径上的mex,那么这个数要满足什么条件?
- 1 ~ x - 1的数都必须出现过.
- x必须没出现过。
现在我们要最大化x,那么也就意味着我们要找到一条路径使得这个都出现过的前缀尽可能长。
第二个条件可以忽略,因为如果第1个条件满足,而第2个条件却不满足,意味着我们可以把x至少扩大1位,因为要求最大值,所以扩大肯定最优,因此我们肯定会扩大到不能扩大为止。
由此我们可以发现,x是满足可二分性的。
考虑在线段树上维护这个问题,区间\([l, r]\)表示值域区间为\([l, r]\)的点全都连接在一起形成的一条路径。
如果有这条路径,那么就存下左右端点,否则就不存。
那么考虑如何合并2个区间\([l, mid], [mid + 1, r]\)。
从4个端点中任取2个作为新的端点,如果另外2个端点均在这2个端点构成的路径上,那么我们就找到了一条新路径满足值域属于\([l, r]\)的点全在路径上。
这样做的正确性(最优性)依赖于权值两两不同,因此我们的选择是唯一的,因此肯定是最优选择。
有一个很简洁的式子可以用于判断一个点是否在某条路径上:
如果\(x\)在链\((u, v)\)上,设\(LCA(u, v) = y\)那么有:
\]
于是维护好之后剩下的事情就是在线段树上查询(二分)了。
#include<bits/stdc++.h>
using namespace std;
#define R register int
#define LL long long
#define AC 201000
#define ac 802000
#define h(x) ((x) << 1)
int n, m;
int Head[AC], date[AC], Next[AC], tot;
int v[AC], fa[AC], s[AC];//s[i]表示权值为i的是哪个,只需要在最开始用就可以了
inline int read()
{
int x = 0;char c = getchar();
while(c > '9' || c < '0') c = getchar();
while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
return x;
}
inline void add(int f, int w)
{date[++ tot] = w, Next[tot] = Head[f], Head[f] = tot;}
#define cal(x, y) ((dep[x] < dep[y]) ? x : y)//返回x, y中dep最小的那个
#define maxn 401000
int st[maxn][20], ss[maxn], dep[AC], t[maxn], p[maxn], first[AC], top;
struct ST_form{
void dfs(int x)//求遍历的数组 + dep
{
ss[++ top] = x, first[x] = top;
for(R i = Head[x]; i; i = Next[i])
{
int now = date[i];
dep[now] = dep[x] + 1, dfs(now), ss[++ top] = x;//返回的时候也要加
}
}
void build()
{
dep[1] = 1, dfs(1);
int tmp = 1, cnt = 0;
for(R i = 1; i <= top; i ++)
{
if(i == (tmp << 1)) tmp <<= 1, ++ cnt;
p[i] = tmp, t[i] = cnt;
}
for(R i = 1; i <= top; i ++) st[i][0] = ss[i];
tmp = 1;
for(R i = 1; i <= 18; i ++)
{
for(R j = 1; j <= top; j ++)
st[j][i] = cal(st[j][i - 1], st[min(j + tmp, top)][i - 1]);
tmp <<= 1;
}
}
inline int LCA(int x, int y)//st表求LCA
{
int l = first[x], r = first[y];
if(l > r) swap(l, r);//不一定按顺序的
int len = r - l + 1;
return cal(st[l][t[len]], st[r - p[len] + 1][t[len]]);
}
}T;
struct node{
int lx, rx;
}tree[ac], go;
#define update(x) tree[x] = merge(tree[x << 1], tree[(x << 1) + 1]);
struct seg_tree{
bool check(int x, int u, int v)//检查x是否在路径(u, v)上
{
int y = T.LCA(u, v);
return ((T.LCA(u, x) == x || T.LCA(v, x) == x) && T.LCA(x, y) == y);
}
inline node merge(node ll, node rr)
{
if(!ll.lx || !rr.lx) return (node){0, 0};
int s[5] = {ll.lx, ll.rx, rr.lx, rr.rx};
for(R i = 0; i < 4; i ++)
for(R j = i + 1; j < 4; j ++)
{
bool flag = true;
if(s[i] == s[j]) continue;
for(R k = 0; k < 4; k ++)
if(k != i && k != j && !check(s[k], s[i], s[j])) {flag = false; break;}
if(flag) return (node){s[i], s[j]};
}
return (node){0, 0};
}
void build(int x, int l, int r)
{
if(l == r) {tree[x].lx = tree[x].rx = s[l]; return ;}
int mid = (l + r) >> 1;
build(h(x), l, mid), build(h(x) + 1, mid + 1, r);
update(x);
}
void change(int x, int l, int r, int w)
{
if(l == r) {tree[x].lx = tree[x].rx = w; return ;}
int mid = (l + r) >> 1;
if(v[w] <= mid) change(h(x), l, mid, w);
else change(h(x) + 1, mid + 1, r, w);
update(x);
}
int find(int x, int l, int r)
{
int mid = (l + r) >> 1;
if(l == r)
{
node tmp = go.lx ? merge(go, tree[x]) : tree[x];
if(tmp.lx) {go = tmp; return 1;}
return 0;
}
if(!tree[h(x)].lx) return find(h(x), l, mid);
else
{
node tmp = go.lx ? merge(go, tree[h(x)]) : tree[h(x)];
if(!tmp.lx) return find(h(x), l, mid);
else {go = tmp; return mid - l + 1 + find(h(x) + 1, mid + 1, r);}
}
}
}T1;
void pre()
{
n = read();
for(R i = 1; i <= n; i ++) v[i] = read() + 1, s[v[i]] = i;
for(R i = 2; i <= n; i ++) fa[i] = read(), add(fa[i], i);
}
void work()
{
m = read();
for(R i = 1; i <= m; i ++)
{
int opt = read();
if(opt == 1)
{
int x = read(), y = read();
swap(v[x], v[y]);
T1.change(1, 1, n, x), T1.change(1, 1, n, y);
}
else go.lx = go.rx = 0, printf("%d\n", T1.find(1, 1, n));//因为向右平移了1,所以要移回来,+ 1 - 1就恰好抵消了
}
}
int main()
{
// freopen("in.in", "r", stdin);
pre();
T.build();
T1.build(1, 1, n);
work();
// fclose(stdin);
return 0;
}
CF1083C Max Mex 线段树的更多相关文章
- Codeforces 1083C Max Mex [线段树]
洛谷 Codeforces 思路 很容易发现答案满足单调性,可以二分答案. 接下来询问就转换成判断前缀点集是否能组成一条链. 我最初的想法:找到点集的直径,判断直径是否覆盖了所有点,需要用到树套树,复 ...
- HDU-4747 Mex 线段树
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4747 题意:求一个数列中,所有mex(L,R)的和. 注意到mex是单调不降的,那么首先预处理出mex ...
- [置顶] hdu4747 Mex 线段树
题意:给你一个序列,让你求出对于所有区间<i, j>的mex和,mex表示该区间没有出现过的最小的整数. 思路:从时限和点数就可以看出是线段树,并且我们可以枚举左端点i, 然后求出所有左端 ...
- hdu 4747 mex 线段树+思维
http://acm.hdu.edu.cn/showproblem.php?pid=4747 题意: 我们定义mex(l,r)表示一个序列a[l]....a[r]中没有出现过得最小的非负整数, 然后我 ...
- [CF1083C]Max Mex
题目大意:有一棵$n(n\leqslant2\times10^5)$个点的树,每个点有点权,所有的点权构成了$0\sim n-1$的排列.$q(q\leqslant2\times10^5)$次操作,操 ...
- BZOJ.3585.mex(线段树)
题目链接 题意:多次求区间\(mex\). 考虑\([1,i]\)的\(mex[i]\),显然是单调的 而对于\([l,r]\)与\([l+1,r]\),如果\(nxt[a[l]]>r\),那么 ...
- bzoj 3585 mex - 线段树 - 分块 - 莫队算法
Description 有一个长度为n的数组{a1,a2,...,an}.m次询问,每次询问一个区间内最小没有出现过的自然数. Input 第一行n,m. 第二行为n个数. 从第三行开始,每行一个询问 ...
- 【bzoj3585】mex 线段树 mex,sg
Description 有一个长度为n的数组{a1,a2,…,an}.m次询问,每次询问一个区间内最小没有出现过的自然数. Input 第一行n,m. 第二行为n个数. 从第三行开始,每行一个询问l, ...
- hdu 4747 Mex( 线段树? 不,区间处理就行(dp?))
Mex Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 65535/65535 K (Java/Others)Total Submis ...
随机推荐
- Ubuntu 16.04.4 LTS环境中php7.0使用mysqli失败,数据库无法访问
环境: Ubuntu 16.04.4 LTS php7.0 输入命令php -m,显示如下,发现没有mysqli模块 输入命令sudo apt install php-mysql,安装mysql模块, ...
- 关于matlab向文件写入数据的方法——留着备用
MATLAB数据采集的时候,往往需要把得到的数据保存下来. fid = fopen(文件名,‘打开方式’): 说明:fid用于存储文件句柄值,如果fid>0,这说明文件打开成功.打开方式有如下选 ...
- ovs源码阅读--netlink使用
netlink netlink socket是一种用于用户态进程和内核态进程之间的通信机制.它通过为内核模块提供一组特殊的API,并为用户程序提供了一组标准的socket接口的方式,实现了全双工的通讯 ...
- 打包一个传统的ASP.NET web app作为Docker镜像
(1)针对NerdDinner应用的Dockerfile内容如下 PS E:\DockeronWindows\Chapter02\ch02-nerd-dinner> cat .\Dockerfi ...
- 10.openldap备份与恢复
备份方式 一.使用slapcat指令备份 使用slapcat备份后的数据 经过相关无用条目处理,即可实现数据上的条目备份 备份指令如下 #备份 #slapcat -v -l openldap-back ...
- Beta冲刺第二周王者荣耀交流协会第五次会议
1.立会照片 成员:王超,高远博,冉华,王磊,王玉玲,任思佳,袁玥全部到齐. master:王磊 2.时间跨度: 2017年11月21日 15:00 — 15:17,总计17分钟. 3.地点: 一食堂 ...
- js中模拟a标签的点击事件
var a = document.createElement('a'); a.target = "_blank"; a.href = "personal"; a ...
- JSON toBean Timestamp To Date 时间戳转日期
时间戳格式的时间从json转为date时 配置: import java.util.Date; import net.sf.ezmorph.object.AbstractObjectMorpher; ...
- 奔跑吧DKY——团队Scrum冲刺阶段-Day 5
今日完成任务 谭鑫:继续解决背景音乐的问题,修改游戏中的bug. 黄宇塘:背景图片需重做,开始制作人物图片和背景图. 赵晓海:制作人物图及背景图. 方艺雯:制作人物图,编写博客. 王禹涵:继续解决背景 ...
- C++:构造函数2——拷贝构造函数
前言:拷贝构造函数是C++中的重点之一,在这里对其知识进行一个简单的总结. 一.什么是拷贝构造函数 在C++中,对于内置类型的变量来说,在其创建的过程中用同类型的另一个变量来初始化它是完全可以的,如 ...