首页 技术 正文
技术 2022年11月23日
0 收藏 825 点赞 2,840 浏览 2529 个字

E – Company

思路:

首先,求出每个点的dfs序

然后求一些点的公共lca, 就是求lca(u, v), 其中u是dfs序最大的点, v是dfs序最小的大点

证明: 假设o是这些点的公共lca, in[o]是o的入时间戳, out[o]是o的出时间戳,

那么对于任意一点x, in[o] <= in[v] <= in[x] <= in[u] <= out[o]

所以o是x的祖先

所以o是这些点的公共lca

所以只要求出每段区间的dfs序最大的和最小的, 删除的要么是最大的, 要么是最小的

可以删除后分成两段区间来考虑或者再求一下次大的和次小的来考虑

求次大次小可以用运算符重载写线段树, 要方便很多

代码:

#pragma GCC optimize(2)
#pragma GCC optimize(3)
#pragma GCC optimize(4)
#include<bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define pi acos(-1.0)
#define LL long long
//#define mp make_pair
#define pb push_back
#define ls rt<<1, l, m
#define rs rt<<1|1, m+1, r
#define ULL unsigned LL
#define pll pair<LL, LL>
#define pli pair<LL, int>
#define pii pair<int, int>
#define piii pair<pii, int>
#define mem(a, b) memset(a, b, sizeof(a))
#define fio ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
#define fopen freopen("in.txt", "r", stdin);freopen("out.txt", "w", stout);
//headconst int N = 1e5 + ;
vector<int> g[N];
int anc[N][], deep[N], dfn[N], tot = ;
struct node {
int mx, mn, _mx, _mn;
node operator ^ (const node & rhs) {
node res = rhs;
if(dfn[mx] >= dfn[res.mx]) {
res._mx = res.mx;
res.mx = mx;
}
else if(dfn[mx] > dfn[res._mx]) {
res._mx = mx;
} if(dfn[mn] <= dfn[res.mn]) {
res._mn = res.mn;
res.mn = mn;
}
else if(dfn[mn] < dfn[res._mn]) {
res._mn = mn;
} if(dfn[_mx] >= dfn[res.mx]) {
res._mx = res.mx;
res.mx = _mx;
}
else if(dfn[_mx] > dfn[res._mx]) {
res._mx = _mx;
} if(dfn[_mn] <= dfn[res.mn]) {
res._mn = res.mn;
res.mn = _mn;
}
else if(dfn[_mn] < dfn[res._mn]) {
res._mn = _mn;
}
return res;
}
}tree[N<<];
void dfs(int u, int o) {
anc[u][] = o;
for (int i = ; i < ; i++) anc[u][i] = anc[anc[u][i-]][i-];
deep[u] = deep[o] + ;
dfn[u] = ++tot;
for (int v : g[u]) {
if(v != o) {
dfs(v, u);
}
}
}
int lca(int u, int v) {
if(deep[u] < deep[v]) swap(u, v);
for (int i = ; i >= ; i--) if(deep[anc[u][i]] >= deep[v]) u = anc[u][i];
if(u == v) return u;
for (int i = ; i >= ; i--) if(anc[u][i] != anc[v][i]) u = anc[u][i], v = anc[v][i];
return anc[u][];
}
void push_up(int rt) {
tree[rt] = tree[rt<<] ^ tree[rt<<|];
}
void build(int rt, int l, int r) {
if(l == r) {
tree[rt] = node{l, l, , };
return ;
}
int m = l+r >> ;
build(ls);
build(rs);
push_up(rt);
}
node query(int L, int R, int rt, int l, int r) {
if(L <= l && r <= R) return tree[rt];
int m = l+r >> ;
node res = {, , , };
if(L <= m) res = query(L, R, ls);
if(R > m) {
if(res.mn == ) res = query(L, R, rs);
else res = res ^ query(L, R, rs);
}
return res;
}
int main() {
int n, p, q, l, r;
scanf("%d %d", &n, &q);
for (int i = ; i <= n; i++) {
scanf("%d", &p);
g[p].pb(i);
}
dfs(, );
dfn[] = ;
build(, , n);
for (int i = ; i <= q; i++) {
scanf("%d %d", &l, &r);
node tmp = query(l, r, , , n);
int l1 = lca(tmp.mx, tmp._mn), l2 = lca(tmp._mx, tmp.mn);
// cout << tmp.mx << " " << tmp._mx << " " << tmp.mn << " " << tmp._mn << endl;
// cout << dfn[tmp.mx] << " " << dfn[tmp._mx] << " " << dfn[tmp.mn] << " " << dfn[tmp._mn] << endl;
// cout << l1 << " " << l2 << endl;
if(deep[l1] < deep[l2]) printf("%d %d\n", tmp.mx, deep[l2]-);
else printf("%d %d\n", tmp.mn, deep[l1]-);
}
return ;
}
相关推荐
python开发_常用的python模块及安装方法
adodb:我们领导推荐的数据库连接组件bsddb3:BerkeleyDB的连接组件Cheetah-1.0:我比较喜欢这个版本的cheeta…
日期:2022-11-24 点赞:878 阅读:8,994
Educational Codeforces Round 11 C. Hard Process 二分
C. Hard Process题目连接:http://www.codeforces.com/contest/660/problem/CDes…
日期:2022-11-24 点赞:807 阅读:5,507
下载Ubuntn 17.04 内核源代码
zengkefu@server1:/usr/src$ uname -aLinux server1 4.10.0-19-generic #21…
日期:2022-11-24 点赞:569 阅读:6,350
可用Active Desktop Calendar V7.86 注册码序列号
可用Active Desktop Calendar V7.86 注册码序列号Name: www.greendown.cn Code: &nb…
日期:2022-11-24 点赞:733 阅读:6,135
Android调用系统相机、自定义相机、处理大图片
Android调用系统相机和自定义相机实例本博文主要是介绍了android上使用相机进行拍照并显示的两种方式,并且由于涉及到要把拍到的照片显…
日期:2022-11-24 点赞:512 阅读:7,768
Struts的使用
一、Struts2的获取  Struts的官方网站为:http://struts.apache.org/  下载完Struts2的jar包,…
日期:2022-11-24 点赞:671 阅读:4,845