首页 > 其他分享 >P10466的题解

P10466的题解

时间:2024-05-17 11:51:34浏览次数:18  
标签:rt cnt ch int 题解 tree xx P10466

(一)

出门左转 P3369

只需要同时记录原本属于哪一位即可。

这里给出 Splay 做法。

(二)

AC 代码。

建议自己打一遍巩固印象。虽然我是直接拉过来的。

#include<bits/stdc++.h>
#define int long long
using namespace std;
inline int read(){
  int x=0,f=1;char c=getchar();
  while(c<'0'||c>'9'){if(c=='-')f=-f;c=getchar();}
  while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
  return x*f; 
}
int n,rt,cnt;
struct node{
	int msiz,tsiz,val,ch[2],f,id;
}tree[1000010];
void maintain(int o){
	if(!o)return;
	tree[o].tsiz=tree[o].msiz;
	if(tree[o].ch[0])tree[o].tsiz+=tree[tree[o].ch[0]].tsiz;
	if(tree[o].ch[1])tree[o].tsiz+=tree[tree[o].ch[1]].tsiz;
}
int get(int o){
	return tree[tree[o].f].ch[1]==o;
}
void clear(int o){
	tree[o].ch[0]=tree[o].ch[1]=tree[o].f=tree[o].msiz=tree[o].tsiz=0;
}
int find_pre(){
	int o=tree[rt].ch[0];
	while(tree[o].ch[1])o=tree[o].ch[1];
	return o;
}
int find_nxt(){
	int o=tree[rt].ch[1];
	while(tree[o].ch[0])o=tree[o].ch[0];
	return o;
}
void rotate(int o){
	int fa=tree[o].f,ffa=tree[fa].f,pos=get(o);
	tree[fa].ch[pos]=tree[o].ch[pos^1];
	tree[tree[fa].ch[pos]].f=fa;
	tree[o].ch[pos^1]=fa;
	tree[fa].f=o;
	tree[o].f=ffa;
	if(ffa)tree[ffa].ch[tree[ffa].ch[1]==fa]=o;
	maintain(fa);
	maintain(o);
}
void splay(int o){
	for(int fa;fa=tree[o].f;rotate(o))
		if(tree[fa].f)rotate((get(o)==get(fa))?fa:o);
	rt=o;
}
void insert(int x,int id){
	if(!rt){
		cnt++;
		rt=cnt;
		tree[cnt].ch[0]=tree[cnt].ch[1]=0;
		tree[cnt].f=0;
		tree[cnt].msiz=tree[cnt].tsiz=1;
		tree[cnt].val=x;
		tree[cnt].id=id;
		return;
	}
	int o=rt,fa=0;
	while(1){
		if(x==tree[o].val){
			tree[o].msiz++;
			maintain(o);
			maintain(fa);
			splay(o);
			break;
		}
		fa=o;
		o=tree[o].ch[x>tree[o].val];
		if(!o){
			cnt++;
			tree[cnt].ch[0]=tree[cnt].ch[1]=0;
			tree[cnt].msiz=tree[cnt].tsiz=1;
			tree[cnt].val=x;
			tree[cnt].f=fa;
			tree[cnt].id=id;
			tree[fa].ch[tree[fa].val<x]=cnt;
			maintain(fa);
			splay(cnt);
			break;
		}
	}
}
int find_rank(int x){
	int ans=0,o=rt;
	while(1){
		if(x<tree[o].val)o=tree[o].ch[0];
		else{
			if(tree[o].ch[0])ans+=tree[tree[o].ch[0]].tsiz;
			if(x==tree[o].val){
				splay(o);
				return ans+1;
			}
			ans+=tree[o].msiz;
			o=tree[o].ch[1];
		}
	}
}
void remove(int x){
	int r=find_rank(x);
	if(tree[rt].msiz>1){
		tree[rt].msiz--;
		maintain(rt);
		return;
	}
	if(!tree[rt].ch[0]&&!tree[rt].ch[1]){
		clear(rt);
		rt=0;
	}
	else if(!tree[rt].ch[0]){
		int xx=rt;
		rt=tree[rt].ch[1];
		tree[rt].f=0;
		clear(xx);
	}
	else if(!tree[rt].ch[1]){
		int xx=rt;
		rt=tree[rt].ch[0];
		tree[rt].f=0;
		clear(xx);
	}
	else{
		int pos=find_pre(),xx=rt;
		splay(pos);
		tree[rt].ch[1]=tree[xx].ch[1];
		tree[tree[xx].ch[1]].f=rt;
		clear(xx);
		maintain(rt);
	}
}
node find_val(int x){
	int o=rt;
	while(1){
		if(tree[o].ch[0]&&tree[tree[o].ch[0]].tsiz>=x)o=tree[o].ch[0];
		else{
			int s=(tree[o].ch[0]?tree[tree[o].ch[0]].tsiz:0)+tree[o].msiz;
			if(x<=s)return tree[o];
			x-=s;
			o=tree[o].ch[1];
		}
	}
}
signed main(){
	freopen("in.txt","r",stdin);
	freopen("out.txt","w",stdout);
	n=read();
	for(int i=1;i<=n;i++){
		int x=read();
		insert(x,i);
		if(i==1)continue;
		int rk=find_rank(x);
		if(rk==1){
			node t=find_val(2);
			printf("%lld %lld\n",t.val-x,t.id);
		}
		else if(rk==i){
			node t=find_val(i-1);
			printf("%lld %lld\n",x-t.val,t.id);
		}
		else{
			node t1=find_val(rk-1),t2=find_val(rk+1);
			if(x-t1.val<=t2.val-x)printf("%lld %lld\n",x-t1.val,t1.id);
			else printf("%lld %lld\n",t2.val-x,t2.id);
		}
	}
    return 0;
}

标签:rt,cnt,ch,int,题解,tree,xx,P10466
From: https://www.cnblogs.com/Jh763878/p/18197577

相关文章

  • Ollama的常见问题解答(FAQ)
     1如何更新Ollama? 在macOS和Windows上,Ollama会自动检查更新。只需点击任务栏或菜单栏图标,然后选择“重启以更新”来应用新版本。 在Linux上,需要重新运行安装脚本来升级。 2如何查看日志? 查阅特定的故障排查文档来了解如何查看和使用日志。 3我的GPU是否兼容Ollama? ......
  • 题解:CF1954F Unique Strings
    link计数类*3100首次独立过纪念版题解。首先我们考虑一个去重的问题。貌似针对循环同构去重的问题,只能从循环节上入手。那么我们考虑设\(dp(d)\)为最小循环节长度恰好为\(d\)不同方案数个数,则答案为:\[\sum_{d=1}^ndp(d)=\sum_{d|n}\frac{dp(d)}{d}\]这似乎是一条可行......
  • 2024 jscpc B题 Area of the Devil 题解
    题目链接:AreaoftheDevil算不在题目说的区域内的面积,直接算是比较麻烦的,这里给一个朋友直接算画的图,其实画出区域以后也算好算,当然官解提到的容斥去算更好写。一共有五个空余的区域,我们考虑这五个区域怎么计算,图一是直接画出的所有区域的并集,图二则是五角星处于边界情况时,图......
  • [ARC149D] 的题解
    思路很巧妙,首先,很明显,数轴上关于原点对称的一个点对,不论移动了多少次,都任然是对称的。再看眼数据范围,发现其实点分布的比较紧,考虑直接将所有点看做一个整体(数轴上一个线段),然后依次移动。关键的是,若这个整体横跨了原点的话,那么在原点的点就有答案了,对于剩下的部分,设在正半轴、负......
  • Codeforces 1004B Sonya and Exhibition 题解
    题目简述让你构造一个长度为$n$的$01$字符串。一个区间的价值为其中$0$的数量乘以$1$的数量,给出$m$个区间,最大化它们的价值之和。题目分析设区间$[l,r]$中$1$有$x$个,$0$有$y$个,当$x$和$y$最接近的时候,$x\timesy$最大,此结论可以用二次函数进行证明。......
  • AT_arc042_c的题解
    (一)非常妙的DP题,可惜被翻译毁了。题意:你有一堆零食,每个零食有两个值\(a_i\)和\(b_i\)。要求选出集合\(S\),使\(\sum_{i\inS}a_i-\min_{i\inS}a_i\lep\),求最大的\(\sum_{i\inS}b_i\)。一眼DP。先将\(a_i\)从小到大排序,每次循环更新\(dp_0\)的值为\(\max......
  • P10447 最短 Hamilton 路径 题解
    P10447最短Hamilton路径题解题目传送门题意:给定一张有向带权图(\(n\le20\))求点\(0\)到点\(n-1\)的最短哈密顿路径。这是一道状压DP模板题。在状态压缩DP中,我们将一个集合压成一个二进制数。设\(f_{u,i}\)为已经走了集合\(u\)中的节点,且现在在点\(i\)的最短......
  • CF1886E 题解
    CF1886E思路观察发现每个项目只与程序员数量和最小值有关,所以每个项目对应能力值连续的程序员最优。项目数\(m\le20\),状压。设\(dp_{i,s}\)为前\(i\)个程序员匹配的项目状态为\(s\)是否可行,无法接受。交换维度,改为\(dp_s\)表示状态\(s\)能与前缀\([1,i]\)匹配的......
  • 旅行 题解
    题目链接。题意简述给定一张有向图,求从点\(A\)走到点\(B\)的一条路径,这条路径满足:经过的边的权值总和是\(P\)的倍数。在满足条件\(1\)的情况下,经过的边的权值总和最小。题目分析本题可以使用分层图最短路来解决。仿照动态规划的思想,定义\(f_{x,y}\)表示从节点......
  • echarts图由于容器隐藏导致图表不显示问题解决办法
    开发过程中常常会遇到echarts图由于容器隐藏导致图表不显示问题,最简单的办法就是给容器元素加上宽度和高度容器加上固定的宽度和高度<divid="res"style="height:450px;width:1200px"></div>然而在实际开发中某些场景下,要求图表宽度100%显示,而计算容器的宽度有时又会十分的麻......