首页 > 其他分享 >【题解】CF1062E Company

【题解】CF1062E Company

时间:2023-05-25 15:33:29浏览次数:52  
标签:ch return int 题解 Company LCA mid CF1062E ans

传送门

先考虑如何求解区间 LCA


假设要我们求 \(8 \sim 11\) 的 LCA。

那么显然它们的 LCA 等效于 8 和 11 的 LCA。

发现 8 和 11 刚好是 “最左” 和 “最右” 的两个顶点,感性理解一下,就可以得出一个结论:

区间 LCA 和 dfs 序 最小的 和 最大的 的LCA 是等效的。(这个结论应该还是蛮经典的)

那么问题就转化成求区间最值,线段树基操。

回到这题,由于上述结论,删去的点显然要么是 dfs 序 最大的, 要么是 dfs 序 最小的,两种情况都做一下即可。

#include<bits/stdc++.h>
#define ls u << 1
#define rs u << 1 | 1
using namespace std;
const int N = 1e5 + 67;
int read(){
	int x = 0, f = 1; char ch = getchar();
	while(ch < '0' || ch > '9'){if(ch == '-') f = -f; ch = getchar();}
	while(ch >= '0' && ch <= '9'){x = (x << 1) + (x << 3) + (ch ^ 48); ch = getchar();}
	return x * f;
}
int n, q, tot, lg, cnt;
int Head[N], to[N << 1], Next[N << 1];
int f[N][20], dfn[N], id[N], d[N];
int minn[N << 2], maxn[N << 2]; 
void add(int u, int v){
	to[++tot] = v, Next[tot] = Head[u], Head[u] = tot;
}
void dfs(int x, int fa){
	d[x] = d[fa] + 1, dfn[x] = ++cnt, id[cnt] = x, f[x][0] = fa;
	for(int i = Head[x]; i; i = Next[i]){
		int y = to[i]; if(y == fa) continue;
		dfs(y, x);
	}
}
void pushup(int u){
	minn[u] = min(minn[ls], minn[rs]);
	maxn[u] = max(maxn[ls], maxn[rs]);
}
void build(int u, int l, int r){
	if(l == r) return minn[u] = maxn[u] = dfn[l], void();
	int mid = (l + r) >> 1;
	build(ls, l, mid), build(rs, mid + 1, r);
	pushup(u);
}
int querymx(int u, int l, int r, int L, int R){
	if(l > R || r < L) return 0;
	if(L <= l && r <= R) return maxn[u];
	int mid = (l + r) >> 1, ans = 0;
	if(L <= mid) ans = max(ans, querymx(ls, l, mid, L, R));
	if(R > mid) ans = max(ans, querymx(rs, mid + 1, r, L, R));
	return ans;
}
int querymn(int u, int l, int r, int L, int R){
	if(l > R || r < L) return 1e9;
	if(L <= l && r <= R) return minn[u];
	int mid = (l + r) >> 1, ans = 1e9;
	if(L <= mid) ans = min(ans, querymn(ls, l, mid, L, R));
	if(R > mid) ans = min(ans, querymn(rs, mid + 1, r, L, R));
	return ans;
}
int lca(int x, int y){
	if(d[x] < d[y]) swap(x, y);
	for(int i = lg; ~i; --i) if(d[f[x][i]] >= d[y]) x = f[x][i];
	if(x == y) return x;
	for(int i = lg; ~i; --i) if(f[x][i] != f[y][i]) x = f[x][i], y = f[y][i];
	return f[x][0];
}
int main(){
	n = read(), q = read(), lg = log2(n);
	for(int i = 2; i <= n; ++i){
		int v = read();
		add(i, v), add(v, i);
	}
	dfs(1, 0);
	build(1, 1, n);
	for(int i = 1; i <= lg; ++i)
		for(int j = 1; j <= n; ++j)
			f[j][i] = f[f[j][i - 1]][i - 1];
	while(q--){
		int l = read(), r = read();
		int L = querymn(1, 1, n, l, r), R = querymx(1, 1, n, l, r);
		L = id[L], R = id[R];
		int LL = min(querymn(1, 1, n, l, L - 1), querymn(1, 1, n, L + 1, r));
		int RR = max(querymx(1, 1, n, l, R - 1), querymx(1, 1, n, R + 1, r));
		LL = id[LL], RR = id[RR];
		int t1 = lca(L, RR), t2 = lca(LL, R);
		if(d[t1] > d[t2]) printf("%d %d\n", R, d[t1] - 1);
		else printf("%d %d\n", L, d[t2] - 1);
	} 
	return 0;
} 

标签:ch,return,int,题解,Company,LCA,mid,CF1062E,ans
From: https://www.cnblogs.com/jiangchen4122/p/17431438.html

相关文章

  • 题解(教主的魔法)P2801
    题目教主的魔法题目描述教主最近学会了一种神奇的魔法,能够使人长高。于是他准备演示给XMYZ信息组每个英雄看。于是$N$个英雄们又一次聚集在了一起,这次他们排成了一列,被编号为$1,2,\ldots,N$。每个人的身高一开始都是不超过$1000$的正整数。教主的魔法每次可以把闭区......
  • 常见问题解决 --- 安卓中一个类中的匿名类和另一个类中的匿名类无法相互传值
      runOnUiThread(newRunnable(){@Overridepublicvoidrun(){//在UI线程中执行的主代码textView.setText("Hello,world!");}});将上面更新主ui放置在匿名内部类的回调方法里即可传值给属性。......
  • AtCoder Beginner Contest 302 H. Ball Collector 题解
    AtCoderBeginnerContest302H.BallCollector题意跳过。可以视作将\(a_i,b_i\)之间连了一条边,然后\(a_i,b_i\)之间只能选一个等价于对于一条边只能选择其一个端点。那么对于只包含树的联通块而言,如果都选择儿子节点,那么会有一个根节点无法被选择上;而对于包含至少一个......
  • NOIP2014普及组试题题解
    1.珠心算测验代码:#include<bits/stdc++.h>#definelllonglongusingnamespacestd;constintN=2e4+39+7;intmp[N],n,a[N],ans=0;intmain(){ cin>>n; for(inti=1;i<=n;i++)cin>>a[i]; for(inti=1;i<=n;i++){ for(intj=1;j<=n;j++)......
  • 常见问题解决 --- Failed to build android app at server - class file for android.
    问题原因  这个错误主要是LocalBroadcastManager这个类被弃用了,而在库或者sdk中使用到了。解决办法build.gradle文件中添加implementation'com.android.support:support-v4:30.4.1'gradle.properties添加android.enableJetifier=true......
  • abc260_f Find 4-cycle 题解
    Find4-cycle题意有一个\(s+t\)个点\(m\)条边的简单无向图\(G\)。点标号为\(1\cdotss+t\),边标号为\(1\cdotsm\)。第\(i\)条边连接点\(u_i\)和\(v_i\)。如果\(G\)中包含一个大小为\(4\)的简单环,选择任意一个并按任意顺序输出环上的\(4\)个点。若不存......
  • abc260_e At Least One 题解
    AtLeastOne题意给定一个整数\(m\)和\(n\)对数\((a_i,b_i)\),我们定义一个\(f(x)\)函数表示满足以下要求的整数序列数量:整数序列为\((1,2,3\cdotsm)\)的一个子段且序列长度为\(x\)。对于\(1\leqslanti\leqslantn\),满足\(a_i\)或者\(b_i\)在整数序列......
  • Element 表格固定列横向滚动条无法拖动的问题解决
    在Element-UI中,当对表格列进行固定后,底部的横向滚动条就无法拖动了,主要的问题就是固定区域盖住了横向滚动条。方案一:修改el-table__body-wrapper样式的层级,随便设个层级就可::v-deep.el-table__body-wrapper{z-index:2}//加了fixed之后失效::v-deep.el-table__fi......
  • abc273_e Notebook 题解
    Notebook题意有\(q\)次操作。现在你有一个空序列\(a\)和一本\(10^9\)页的笔记本,每页纸上都有一个空序列。每次操作是以下四种中的一种:ADDx,表示在\(a\)的末尾插入一个整数\(x\)。DELETE,表示删除\(a\)的末尾的一个数,如果\(a\)序列为空则什么也不干。SAVEy,表......
  • Game on Paper 题解
    题目传送门一道模拟题。如果每涂一个格子就判断整个矩阵,那时间复杂度显然会炸。我们每涂一个格子,影响的应该只是以这个格子为中心的\(3\times3\)矩阵,判断以这些点为中心的话会不会涂出\(3\times3\)的矩阵即可。Code#include<bits/stdc++.h>#definelllonglong#d......