【替罪羊树-动态标号+线段树】BZOJ3600[没有人的算术]题解

14 篇文章 0 订阅
11 篇文章 0 订阅

题目概述

定义一种数,这种数只有 0 (a,b) ,其中 a,b 是这种数。

定义 < <script type="math/tex" id="MathJax-Element-4"><</script> :1. 0 最小。2. (a,b)<(c,d),c>a 。3. (a,b)<(a,c),c>b

现在有 n 个这种数,刚开始都是 0 。给出 m 个操作:1.将 a[k] 赋值为 (a[l],a[r]) 。2.询问 [l,r] 中最大值的编号,有多个最大值取最小编号。

解题报告

询问是裸的线段树,但是真的按题目意思去比大小的话就会非常复杂,实际上我们可以按照这种数的大小关系将每个数都等价为一个实数,然后我们比较实数大小就行了。

因为每次会产生新的数,所以这是一个动态标号问题。快速确定位置可以用平衡树,但是Treap,Splay等旋转平衡树显然无法快速编号。这样的话我们可以用替罪羊树或者非旋Treap。因为替罪羊树好打就用替罪羊树好了~

解题报告

第一次用Emacs打代码,真的是心累QAQ。

#include<cstdio>
#include<cctype>
#include<algorithm>
#define fr first
#define sc second
#define mp make_pair
using namespace std;
typedef long double DB;
const int maxn=1e5,maxt=6e5;const DB AL=0.75;

int n,te,Tail,MAX[(maxn<<2)+5];
struct node{
    node *son[2];pair<node*,node*> val;DB now;int si;node() {}
    node(node *p,pair<node*,node*> v,DB k=0,int s=1) {son[0]=son[1]=p;val=v;now=k;si=s;}
    inline void Pushup() {si=son[0]->si+1+son[1]->si;}
    inline bool isbad() {return si*AL<max(son[0]->si,son[1]->si);}
}nil(&nil,mp(&nil,&nil),0,0);
typedef node* P_node;typedef pair<P_node,P_node> data;
bool operator < (const data &a,const data &b) {return a.fr->now<b.fr->now||a.fr->now==b.fr->now&&a.sc->now<b.sc->now;}
bool operator == (const data &a,const data &b) {return a.fr==b.fr&&a.sc==b.sc;}
node tem[maxt+5];P_node si=tem,null=&nil,ro=null;
DB sl,sr;P_node ID[maxn+5],que[maxt+5],*sgt;

#define Eoln(x) ((x)==10||(x)==13||(x)==EOF)
inline char readc(){
    static char buf[100000],*l=buf,*r=buf;
    if (l==r) r=(l=buf)+fread(buf,1,100000,stdin);
    if (l==r) return EOF;return *l++;
}
inline int readi(int &x){
    int tot=0,f=1;char ch=readc(),lst='+';
    while (!isdigit(ch)) {if (ch==EOF) return EOF;lst=ch;ch=readc();}
    if (lst=='-') f=-f;
    while (isdigit(ch)) tot=(tot<<3)+(tot<<1)+ch-48,ch=readc();
    return x=tot*f,Eoln(ch);
}
inline char getfst() {char ch=readc();while (ch!='C'&&ch!='Q') ch=readc();return ch;}
#define newnode(v,k) (*si=node(null,v,k),si++)
P_node ins(P_node &p,data k,DB l=0,DB r=1){
    DB m=(l+r)/2;if (p==null) p=newnode(k,m);if (k==p->val) return p;P_node now;
    if (k<p->val) now=ins(p->son[0],k,l,m); else now=ins(p->son[1],k,m,r);
    p->Pushup();if (p->isbad()) sgt=&p,sl=l,sr=r;return now;
}
void Dfs(P_node p){
    if (p==null) return;
    Dfs(p->son[0]);que[++Tail]=p;Dfs(p->son[1]);
}
P_node Build(int L,int R,DB l,DB r){
    if (L>R) return null;int mid=L+(R-L>>1);DB m=(l+r)/2;que[mid]->now=m;
    que[mid]->son[0]=Build(L,mid-1,l,m);que[mid]->son[1]=Build(mid+1,R,m,r);
    que[mid]->Pushup();return que[mid];
}
#define Rebuild Tail=0,Dfs(*sgt),*sgt=Build(1,Tail,sl,sr)
inline P_node Insert(data k) {sgt=0;P_node now=ins(ro,k);if (sgt) Rebuild;return now;}
#define LS (p<<1)
#define RS (p<<1|1)
#define Fix(x,y) if (ID[y]->now>ID[x]->now||ID[y]->now==ID[x]->now&&y<x) x=y
inline void Pushup(int p) {MAX[p]=MAX[LS];Fix(MAX[p],MAX[RS]);}
void Update(int pos,int l=1,int r=n,int p=1){
    if (pos<l||r<pos) return;if (l==r) {MAX[p]=pos;return;}
    int mid=l+(r-l>>1);Update(pos,l,mid,LS);Update(pos,mid+1,r,RS);Pushup(p);
}
inline int Maxer(int x,int y) {Fix(x,y);return x;}
int Ask(int L,int R,int l=1,int r=n,int p=1){
    if (R<l||r<L) return 0;if (L<=l&&r<=R) return MAX[p];
    int mid=l+(r-l>>1);return Maxer(Ask(L,R,l,mid,LS),Ask(L,R,mid+1,r,RS));
}
int main(){
    freopen("program.in","r",stdin);
    freopen("program.out","w",stdout);
    readi(n);ID[0]=null;for (int i=1;i<=n;i++) ID[i]=Insert(mp(null,null)),Update(i);
    for (readi(te);te;te--){
        char td=getfst();int L,R,k;readi(L);readi(R);
        if (td=='C') readi(k),ID[k]=Insert(mp(ID[L],ID[R])),Update(k); else
        printf("%d\n",Ask(L,R));
    }
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值