本blog会讲一些简单的Splay的应用,包括但不局限于

1. Splay 维护数组下标,支持区间reserve操作,解决区间问题

2. Splay 的启发式合并(按元素多少合并)

3. 线段树+Splay 大常数树套树

一、Splay维护区间下标解决区间翻转问题

思想: 对于数组的下标是不可重复的,我们使用平衡树维护下标,利用Splay的splay操作,让区间都在一棵子树内。

然后直接输出这颗子树的维护信息,由于维护的是子树信息,那么父亲的信息一定可以由两个儿子推出。

于是就可以类似于线段树的操作了(用儿子信息维护父亲信息)。

[LuoGu 模板] 文艺平衡树 : 维护区间翻转,输出最终区间形态。

Solution :

考虑splay函数的实质,就是把节点x转到目标节点,并且让Splay树保持BST的性质。

那么显然,对于维护序列下标,其维护的值便是元素对于数组的位置。(一个节点左边儿子子树的位置都小于该节点)

对于一个区间进行翻转,那么事实上就是 把l和r下标交换,l+1和r-1下标交换...

那么如何确定这个区间呢,不妨把[l-1]转到根节点,[r+1]转到根节点的右儿子,那么对于根节点右儿子的左儿子所在子树

就可以描述为[l,r]区间。

然而,对于Splay本身来说是不能出现下标为0的情况的,否则会死循环,在(1)中已经讲的非常明确了,所以我们仍然加入一个哨兵节点,对于所有区间均+1

在代码中的实现是这样的。

需要注意的是,特殊的,对于维护位置(下标)是不会出现重复元素的,那么所有节点cnt的值都是1,就省略了。

最后这颗Splay的中序遍历就是最终区间。

# include <bits/stdc++.h>
# define inf (0x3f3f3f3f)
using namespace std;
const int N=2e5+;
inline int read()
{
int X=,w=; char c=;
while(c<''||c>'') {w|=c=='-';c=getchar();}
while(c>=''&&c<='') X=(X<<)+(X<<)+(c^),c=getchar();
return w?-X:X;
}
void write(int x) {
if (x<) putchar('-'),x=-x;
if (x>) write(x/);
putchar(x%+'');
}
void writeln(int x){
write(x);putchar('\n');
}
struct Splay{
# define ls(x) (t[x][])
# define rs(x) (t[x][])
int t[N][],size[N],val[N],par[N];
bool rev[N];
int root,tot;
Splay(){ root=,tot=;insert(-inf);insert(inf);}
int check (int x) { return rs(par[x])==x;}
void pushup(int x) { size[x]=size[ls(x)]+size[rs(x)]+;}
void rotate(int x) {
int y=par[x],k=check(x);
t[y][k]=t[x][k^]; par[t[x][k^]]=y;
t[x][k^]=y;
t[par[y]][check(y)]=x; par[x]=par[y]; par[y]=x;
pushup(y); pushup(x);
}
void splay(int x,int goal=) {
while (par[x]!=goal) {
int y=par[x],z=par[y];
if (z!=goal) rotate(check(x)==check(y)?y:x);
rotate(x);
}
if (!goal) root=x;
}
void down(int x) {
if (rev[x]) {
swap(ls(x),rs(x));
rev[ls(x)]^=; rev[rs(x)]^=;
rev[x]=;
}
}
void insert(int x){
int cur=root,p=;
while (cur && val[cur]!=x)
p=cur,cur=t[cur][x>val[cur]];
cur=++tot;
if (p) t[p][x>val[p]]=cur;
t[cur][]=t[cur][]=;
val[cur]=x; par[cur]=p;
size[cur]=;
splay(cur);
}
int kth_id(int k) {
int cur=root;
while (true) {
down(cur);
if (t[cur][] && k<=size[ls(cur)]) cur=ls(cur);
else if (k>size[ls(cur)]+) {
k-=size[ls(cur)]+;
cur=rs(cur);
} else return cur;
}
}
void reserve(int l,int r) {
int x=kth_id(l),y=kth_id(r+);
splay(x); splay(y,x);
rev[ls(y)]^=;
}
void print(int x) {
down(x);
if (ls(x)) print(ls(x));
if (val[x]!=-inf&&val[x]!=inf) printf("%d ",val[x]);
if (rs(x)) print(rs(x));
}
}tr;
int main()
{
int n=read(),m=read();
for (int i=;i<=n;i++) tr.insert(i);
for (int i=;i<=m;i++) {
int l=read(),r=read();
tr.reserve(l,r);
}
tr.print(tr.root);
return ;
}

P3391 【模板】文艺平衡树(Splay)

[例题 : 序列终结者]

用Splay维护3个操作

1 L R V : 区间[L,R]每个元素+V

2 L R  :将区间[L,R]翻转

3 L R  :输出区间[L,R]的最小值

Hint : 元素初始值都为 0

Solution:

显然我们可以使用上述方法维护一棵平衡树,还是维护下标,我们发现splay或者rotate一下,树维护下标的性质是不会改变的。

所以仍然可以在节点维护一个加法标记,当翻转前下传即可。

# include<bits/stdc++.h>
# define inf (0x3f3f3f3f)
using namespace std;
const int N=1e5+;
int n,m,a[N];
struct Splay{
# define ls(x) (t[x][])
# define rs(x) (t[x][])
int t[N][],par[N],size[N],val[N],mx[N],add[N];
int root,tot;
bool rev[N];
int check(int x){return rs(par[x])==x; }
void up(int x) {
size[x]=size[ls(x)]+size[rs(x)]+;
mx[x]=max(max(mx[ls(x)],mx[rs(x)]),val[x]);
}
void down(int x) {
int &l=ls(x),&r=rs(x);
if (add[x]) {
if (l) add[l]+=add[x],val[l]+=add[x],mx[l]+=add[x];
if (r) add[r]+=add[x],val[r]+=add[x],mx[r]+=add[x];
}
if (rev[x]) rev[l]^=,rev[r]^=,swap(l,r);
rev[x]=add[x]=;
}
void rotate(int x) {
int y=par[x],k=check(x);
down(x); down(y);
t[y][k]=t[x][k^]; par[t[x][k^]]=y;
t[x][k^]=y; t[par[y]][check(y)]=x;
par[x]=par[y];par[y]=x;
up(y); up(x);
}
void splay(int x,int goal=)
{
while (par[x]!=goal) {
int y=par[x],z=par[y];
if (z!=goal) rotate((check(x)==check(y))?y:x);
rotate(x);
}
if (!goal) root=x;
}
int build(int l,int r,int fa) {
if(l>r) return ;
int x=++tot,mid=(l+r)>>;
par[x]=fa,val[x]=mx[x]=a[mid];
ls(x)=build(l,mid-,x);
rs(x)=build(mid+,r,x);
up(x); return x;
}
int kth_id(int k) {
int cur=root;
while (cur) {
down(cur);
if (t[cur][] && k<=size[ls(cur)]) cur=ls(cur);
else if (k>size[ls(cur)]+) {
k-=size[ls(cur)]+;
cur=rs(cur);
} else return cur;
}
}
int kth_val(int k) {
int p=kth_id(k);
return val[p];
}
void reserve(int l,int r)
{
int x=kth_id(l),y=kth_id(r+);
splay(x); splay(y,x);
rev[ls(y)]^=;
}
int GetMax(int l,int r) {
int x=kth_id(l),y=kth_id(r+);
splay(x); splay(y,x);
return mx[ls(y)];
}
int Add(int l,int r,int d)
{
int x=kth_id(l),y=kth_id(r+);
splay(x); splay(y,x); val[ls(y)]+=d;
add[ls(y)]+=d; mx[ls(y)]+=d;
}
}tr;
int main()
{
scanf("%d%d",&n,&m);
memset(a,,sizeof(a));
tr.mx[]=a[]=a[n+]=-inf;
tr.root=tr.build(,n+,);
while (m--) {
int op,l,r,d;
scanf("%d%d%d",&op,&l,&r);
if (op==) scanf("%d",&d),tr.Add(l,r,d);
else if (op==) tr.reserve(l,r);
else if (op==) printf("%d\n",tr.GetMax(l,r));
}
return ;
}

序列终结者.cpp

、Splay的启发式合并(按元素多少合并)

P3224 [HNOI2012]永无乡

维护两个操作:

一开始所有元素都独立。

1 . 合并x,y

2 . 在节点x连通块内求节点排名为k的元素编号。

Sol : 首先,对于每个元素都维护一棵平衡树,由于我们是数组模拟链表,所以在"指针池"中,只需要记录根是谁。

就可以维护一棵平衡树森林了,我们只要求在平衡树森林中总节点个数不超过"指针池"大小即可。

当合并节点x,y的时候,若以x所在节点的平衡树和y所在节点平衡树合并的时候,按节点个数合并即可,尽可能保证元素多的Splay树不改变,

然后把元素少的Splay树遍历依次,把所有拥有的元素插入到元素多的Splay树中

为了处理方便,我们令第i棵子树根的编号为father(i)

# include <bits/stdc++.h>
using namespace std;
const int N=3e5+;
map<int,int>Hash;
int t[N][],par[N],root[N],size[N],cnt[N],val[N];
int n,m,tot,f[N];
# define ls(x) t[x][]
# define rs(x) t[x][]
inline int read()
{
int X=,w=; char c=;
while(c<''||c>'') {w|=c=='-';c=getchar();}
while(c>=''&&c<='') X=(X<<)+(X<<)+(c^),c=getchar();
return w?-X:X;
}
inline void write(int x)
{
if (x<) putchar('-'),x=-x;
if (x>) write(x/);
putchar(''+x%);
}
int check(int x) { return rs(par[x])==x; }
void up(int x){
size[x]=size[ls(x)]+size[rs(x)]+cnt[x];
}
void rotate(int x){
int y=par[x],k=check(x);
t[y][k]=t[x][k^]; par[t[x][k^]]=y;
t[x][k^]=y; t[par[y]][check(y)]=x;
par[x]=par[y]; par[y]=x;
up(y); up(x);
}
void splay(int x,int goal) {
while (par[x]!=goal) {
int y=par[x],z=par[y];
if (z!=goal) {
if (check(x)==check(y)) rotate(y);
else rotate(x);
}
rotate(x);
}
if (goal<=n) root[goal]=x;
}
void insert(int rt,int x) {
int cur=root[rt],p=rt;
while (cur && val[cur]!=x)
p=cur,cur=t[cur][x>val[cur]];
if (cur) cnt[cur]++;
else {
cur=++tot;
if (p>n) t[p][x>val[p]]=cur;
ls(cur)=rs(cur)=;
par[cur]=p; val[cur]=x;
size[cur]=cnt[cur]=;
}
splay(cur,rt);
}
int kth(int rt,int k) {
int cur=root[rt];
if (k>size[cur]||k<) return -;
while (true) {
if (t[cur][]&&k<=size[ls(cur)]) cur=ls(cur);
else if (k>size[ls(cur)]+cnt[cur]) {
k-=size[ls(cur)]+cnt[cur];
cur=rs(cur);
} else return val[cur];
}
}
int father(int x)
{
if (f[x]==x) return x;
return f[x]=father(f[x]);
}
void dfs(int tr1,int tr2)
{
if (ls(tr1)) dfs(ls(tr1),tr2);
if (rs(tr1)) dfs(rs(tr1),tr2);
insert(tr2,val[tr1]);
}
void merge(int u,int v)
{
int fx=father(u),fy=father(v);
if (size[root[fx]]>size[root[fy]]) swap(fx,fy);
f[fx]=fy;
dfs(root[fx],fy);
}
int main()
{
n=read();m=read();
for (int i=;i<=n;i++) f[i]=i,root[i]=i+n;
for (int i=;i<=n;i++){
int x=read(); Hash[x]=i;
val[i+n]=x; cnt[i+n]=size[i+n]=; par[i+n]=i;
}
tot=n<<;
for (int i=;i<=m;i++) {
int u=read(),v=read();
merge(u,v);
}
int Q=read();
char op[];
while (Q--) {
scanf("%s",op);int x=read(),y=read();
if (op[]=='B') merge(x,y);
else {
int ans=kth(father(x),y);
if (ans==-) write(-),putchar('\n');
else write(Hash[ans]),putchar('\n');
}
}
return ;
}

永无乡.cpp

三. 线段树+Splay 大常数树套树

【模板】二逼平衡树(树套树)

// luogu-judger-enable-o2
// luogu-judger-enable-o2
// luogu-judger-enable-o2
# pragma GCC optimize()
# include <cstdio>
# define il inline
# define Rint register int
using namespace std;
const int inf=;
const int N=5e4+;
int a[N];
int n,m,Lim;
il int max(int a,int b) {
return (a<b)?b:a;
}
il int min(int a,int b) {
return (a>b)?b:a;
}
il int read()
{
int X=,w=; char c=;
while(c<''||c>'') {w|=c=='-';c=getchar();}
while(c>=''&&c<='') X=(X<<)+(X<<)+(c^),c=getchar();
return w?-X:X;
}
il void write(int x)
{
if (x<) putchar('-'),x=-x;
if (x>) write(x/);
putchar(''+x%);
}
il void writeln(int x){write(x);putchar('\n');}
struct SegmentTree{
int l,r,root;
}tr[N<<];
struct Splay{
# define ls(x) (t[x][])
# define rs(x) (t[x][])
int t[N*][],cnt[N*],val[N*],size[N*],par[N*];
int tot;
Splay() { tot=; }
il int check(int x) { return rs(par[x])==x; }
il void up(int x){
size[x]=size[ls(x)]+size[rs(x)]+cnt[x];
}
il void rotate(int x){
int y=par[x],k=check(x);
t[y][k]=t[x][k^]; par[t[x][k^]]=y;
t[x][k^]=y; t[par[y]][check(y)]=x;
par[x]=par[y]; par[y]=x;
up(y); up(x);
}
il void splay(int node,int x,int goal) {
while (par[x]!=goal) {
int y=par[x],z=par[y];
if (z!=goal) {
if (check(x)==check(y)) rotate(y);
else rotate(x);
}
rotate(x);
}
if (!goal) tr[node].root=x;
}
il void insert(int node,int x) {
int cur=tr[node].root,p=;
if (!cur) {
cur=++tot; tr[node].root=cur;
ls(cur)=rs(cur)=;
par[cur]=; val[cur]=x;
size[cur]=cnt[cur]=;
return;
}
while (cur && val[cur]!=x)
p=cur,cur=t[cur][x>val[cur]];
if (cur) cnt[cur]++;
else {
cur=++tot;
if (p) t[p][x>val[p]]=cur;
ls(cur)=rs(cur)=;
par[cur]=p; val[cur]=x;
size[cur]=cnt[cur]=;
}
splay(node,cur,);
}
il void find(int node,int x){
int cur=tr[node].root;
if (!cur) return;
while (t[cur][x>val[cur]] && val[cur]!=x)
cur=t[cur][x>val[cur]];
splay(node,cur,);
}
il int pre_id(int node,int x) {
find(node,x); int cur=tr[node].root;
if (val[cur]<x) return cur;
cur=ls(cur);
while (rs(cur)) cur=rs(cur);
return cur;
}
il int pre_val(int node,int x){
find(node,x); int cur=tr[node].root;
if (val[cur]<x) return val[cur];
cur=ls(cur);
while (rs(cur)) cur=rs(cur);
return val[cur];
}
il int suc_id(int node,int x) {
find(node,x); int cur=tr[node].root;
if (val[cur]>x) return cur;
cur=rs(cur);
while (ls(cur)) cur=ls(cur);
return cur;
}
il int suc_val(int node,int x){
find(node,x); int cur=tr[node].root;
if (val[cur]>x) return val[cur];
cur=rs(cur);
while (ls(cur)) cur=ls(cur);
return val[cur];
}
il void erase(int node,int x){
int last=pre_id(node,x);
int next=suc_id(node,x);
splay(node,next,); splay(node,last,next);
int d=rs(last);
if (cnt[d]>) cnt[d]--,splay(node,d,);
else rs(last)=;
up(last);
}
il int rank(int node,int x) {
find(node,x); int cur=tr[node].root;
if (val[cur]>=x) return size[ls(cur)]-;
else return size[ls(cur)]+cnt[cur]-;
}
# undef ls
# undef rs
}t;
# define lson *x,l,mid
# define rson *x+,mid+,r
# define mid ((l+r)>>)
int Ans;
il void SegBuild(int x,int l,int r)
{
t.insert(x,-inf); t.insert(x,inf);
if (l==r) return;
SegBuild(lson); SegBuild(rson);
}
il void SegInsert(int x,int l,int r,int pos,int val)
{
t.insert(x,val);
if (l==r) return;
if (pos<=mid) SegInsert(lson,pos,val);
else SegInsert(rson,pos,val);
}
il void SegRank(int x,int l,int r,int val,int opl,int opr)
{
if (opl<=l&&r<=opr) {
Ans+=t.rank(x,val);
return;
}
if (opl<=mid) SegRank(lson,val,opl,opr);
if (opr>mid) SegRank(rson,val,opl,opr);
}
il void SegUpdate(int x,int l,int r,int pos,int val)
{
t.erase(x,a[pos]); t.insert(x,val);
if (l==r) { a[pos]=val; return; }
if (pos<=mid) SegUpdate(lson,pos,val);
else SegUpdate(rson,pos,val);
}
il void SegPre(int x,int l,int r,int opl,int opr,int val)
{
if (opl<=l&&r<=opr) {
Ans=max(Ans,t.pre_val(x,val));
return;
}
if (opl<=mid) SegPre(lson,opl,opr,val);
if (opr>mid) SegPre(rson,opl,opr,val);
}
il void SegSuc(int x,int l,int r,int opl,int opr,int val)
{
if (opl<=l&&r<=opr) {
Ans=min(t.suc_val(x,val),Ans);
return;
}
if (opl<=mid) SegSuc(lson,opl,opr,val);
if (opr>mid) SegSuc(rson,opl,opr,val);
}
il int SegKth(int opl,int opr,int k)
{
int l=,r=Lim,ans;
while (l<=r) {
int Mid=(l+r)>>;
Ans=; SegRank(,,n,Mid,opl,opr);
if (Ans+>k) r=Mid-;
else l=Mid+,ans=Mid;
}
return ans;
}
int main()
{
n=read();m=read();
SegBuild(,,n);
for (Rint i=;i<=n;i++) {
a[i]=read(); Lim=max(Lim,a[i]);
SegInsert(,,n,i,a[i]);
}
for (Rint i=;i<=m;i++) {
int op=read(),l=read(),r=read(),k;
if (op==) k=read(),Ans=,SegRank(,,n,k,l,r),writeln(Ans+);
if (op==) k=read(),writeln(SegKth(l,r,k));
if (op==) SegUpdate(,,n,l,r);
if (op==) k=read(),Ans=-inf,SegPre(,,n,l,r,k),writeln(Ans);
if (op==) k=read(),Ans=inf,SegSuc(,,n,l,r,k),writeln(Ans);
}
return ;
}

二逼平衡树(Splay).cpp

# include <bits/stdc++.h>
# define inf ()
using namespace std;
const int N=4e6+;
int n,m,Lim,tot;
int a[N];
struct Treap_Node{
int val,key,cnt,size;
int ch[];
}t[N];
struct Treap {
# define ls t[x].ch[]
# define rs t[x].ch[]
void up(int &x){t[x].size=t[x].cnt+t[ls].size+t[rs].size;}
void rotate(int &x,int d)
{
int son=t[x].ch[d];
t[x].ch[d]=t[son].ch[d^];
t[son].ch[d^]=x;
up(x); up(x=son);
}
void insert(int &x,int val)
{
if (!x) {
x=++tot;
t[x].size=t[x].cnt=;
t[x].val=val;
t[x].key=rand();
return;
}
t[x].size++;
if (t[x].val==val) { t[x].cnt++; return;}
int d=t[x].val<val;
insert(t[x].ch[d],val);
if (t[x].key>t[t[x].ch[d]].key) rotate(x,d);
}
void erase(int &x,int val)
{
if (!x) return;
if (t[x].val==val) {
if (t[x].cnt>) { t[x].cnt--; t[x].size--; return;}
int d=t[ls].key>t[rs].key;
if (ls==||rs==) x=ls+rs;
else rotate(x,d),erase(x,val);
} else t[x].size--,erase(t[x].ch[t[x].val<val],val);
}
inline int rank(int &x,int val){
if (!x) return ;
if (t[x].val==val) return t[ls].size;
if (t[x].val>val) return rank(ls,val);
return t[x].cnt+t[ls].size+rank(rs,val);
}
int find(int &rt,int k)
{
int x=rt;
while () {
if (k<=t[ls].size) x=ls;
else if (k>t[x].cnt+t[ls].size) k-=t[x].cnt+t[ls].size,x=rs;
else return t[x].val;
}
}
inline int pre(int &x,int val)
{
if (!x) return -inf;
if (t[x].val>=val) return pre(ls,val);
return max(pre(rs,val),t[x].val);
}
int nex(int &x,int val)
{
if (!x) return inf;
if (t[x].val<=val) return nex(rs,val);
return min(nex(ls,val),t[x].val);
}
#undef ls
#undef rs
}tr;
struct Segment_Tree{
int l,r,root;
}tree[N];
# define lson (x<<),l,mid
# define rson (x<<)+,mid+,r
# define mid ((l+r)>>)
void SegInsert(int x,int l,int r,int pos,int val)
{
tr.insert(tree[x].root,val);
if (l==r) return;
if (pos<=mid) SegInsert(lson,pos,val);
else SegInsert(rson,pos,val);
}
int SegRank(int x,int l,int r,int ql,int qr,int val)
{
if (ql<=l&&r<=qr) return tr.rank(tree[x].root,val);
int ret=;
if (ql<=mid) ret+=SegRank(lson,ql,qr,val);
if (qr>mid) ret+=SegRank(rson,ql,qr,val);
return ret;
}
void SegUpdate(int x,int l,int r,int pos,int val)
{
tr.insert(tree[x].root,val);
tr.erase(tree[x].root,a[pos]);
if (l==r) { a[pos]=val; return;}
if (pos<=mid) SegUpdate(lson,pos,val);
else SegUpdate(rson,pos,val);
}
int SegPre(int x,int l,int r,int ql,int qr,int val)
{
if (ql<=l&&r<=qr) return tr.pre(tree[x].root,val);
int ret=-inf;
if (ql<=mid) ret=max(ret,SegPre(lson,ql,qr,val));
if (qr>mid) ret=max(ret,SegPre(rson,ql,qr,val));
return ret;
}
int SegSuc(int x,int l,int r,int ql,int qr,int val)
{
if (ql<=l&&r<=qr) return tr.nex(tree[x].root,val);
int ret=inf;
if (ql<=mid) ret=min(ret,SegSuc(lson,ql,qr,val));
if (qr>mid) ret=min(ret,SegSuc(rson,ql,qr,val));
return ret;
}
# undef lson
# undef rson
# undef mid
int SegKth(int l,int r,int k)
{
int L=,R=Lim+,Ans;
while (L<=R) {
int Mid=(L+R)>>;
if (SegRank(,,n,l,r,Mid)+>k) R=Mid-;
else Ans=Mid,L=Mid+;
}
return Ans;
}
int main()
{
scanf("%d%d",&n,&m);
for (int i=;i<=n;i++) {
int t; scanf("%d",&t); Lim=max(Lim,t);
a[i]=t; SegInsert(,,n,i,t);
}
while (m--) {
int op,l,r,k; scanf("%d",&op);
if (op==) scanf("%d%d%d",&l,&r,&k),printf("%d\n",SegRank(,,n,l,r,k)+);
if (op==) scanf("%d%d%d",&l,&r,&k),printf("%d\n",SegKth(l,r,k));
if (op==) scanf("%d%d",&l,&r),SegUpdate(,,n,l,r);
if (op==) scanf("%d%d%d",&l,&r,&k),printf("%d\n",SegPre(,,n,l,r,k));
if (op==) scanf("%d%d%d",&l,&r,&k),printf("%d\n",SegSuc(,,n,l,r,k));
}
return ;
}

二逼平衡树(Treap).cpp

文艺平衡Splay树学习笔记(2)的更多相关文章

  1. BST,Splay平衡树学习笔记

    BST,Splay平衡树学习笔记 1.二叉查找树BST BST是一种二叉树形结构,其特点就在于:每一个非叶子结点的值都大于他的左子树中的任意一个值,并都小于他的右子树中的任意一个值. 2.BST的用处 ...

  2. zkw线段树学习笔记

    zkw线段树学习笔记 今天模拟赛线段树被卡常了,由于我自带常数 \(buff\),所以学了下zkw线段树. 平常的线段树无论是修改还是查询,都是从根开始递归找到区间的,而zkw线段树直接从叶子结点开始 ...

  3. 仙人掌&圆方树学习笔记

    仙人掌&圆方树学习笔记 1.仙人掌 圆方树用来干啥? --处理仙人掌的问题. 仙人掌是啥? (图片来自于\(BZOJ1023\)) --也就是任意一条边只会出现在一个环里面. 当然,如果你的图 ...

  4. 线段树学习笔记(基础&进阶)(一) | P3372 【模板】线段树 1 题解

    什么是线段树 线段树是一棵二叉树,每个结点存储需维护的信息,一般用于处理区间最值.区间和等问题. 线段树的用处 对编号连续的一些点进行修改或者统计操作,修改和统计的复杂度都是 O(log n). 基础 ...

  5. Treap-平衡树学习笔记

    平衡树-Treap学习笔记 最近刚学了Treap 发现这种数据结构真的是--妙啊妙啊~~ 咳咳.... 所以发一发博客,也是为了加深蒟蒻自己的理解 顺便帮助一下各位小伙伴们 切入正题 Treap的结构 ...

  6. JSOI2008 Blue Mary开公司 | 李超线段树学习笔记

    题目链接:戳我 这相当于是一个李超线段树的模板qwqwq,题解就不多说了. 代码如下: #include<iostream> #include<cstdio> #include ...

  7. Splay树学习

    首先给出一论文讲的很好: http://www.docin.com/p-63165342.html http://www.docin.com/p-62465596.html 然后给出模板胡浩大神的模板 ...

  8. Splay伸展树学习笔记

    Splay伸展树 有篇Splay入门必看文章 —— CSDN链接 经典引文 空间效率:O(n) 时间效率:O(log n)插入.查找.删除 创造者:Daniel Sleator 和 Robert Ta ...

  9. 伸展树(Splay)学习笔记

    二叉排序树能够支持多种动态集合操作,它可以被用来表示有序集合,建立索引或优先队列等.因此,在信息学竞赛中,二叉排序树应用非常广泛. 作用于二叉排序树上的基本操作,其时间复杂度均与树的高度成正比,对于一 ...

随机推荐

  1. [转帖]SAP一句话入门:Project System

    SAP一句话入门:Project System http://blog.vsharing.com/MilesForce/A621279.html 这是SAP ERP入门的最后一篇了. 我们这些死跑龙套 ...

  2. [转帖]SAP一句话入门:Production Planning

    SAP一句话入门:Production Planning http://blog.vsharing.com/MilesForce/A617692.html SAP是庞大的,模块是多多的,功能是强大的, ...

  3. @PathVariable

    @PathVariable是用来对指定请求的URL路径里面的变量 eg: Java代码 @RequestMapping(value = "form/{id}/apply", met ...

  4. Windows环境下在IDEA编辑器中spark开发安装步骤

    以下是windows环境下安装spark的过程: 1.安装JDK(version:1.8.0.152) 2.安装scala(version:2.11/2.12) 3.安装spark(version:s ...

  5. centos5 安装redmine

    一.下载依赖包 yum -y install libxslt-devel libyaml-devel libxml2-devel gdbm-devel libffi-devel yum -y inst ...

  6. WPF Image控件的绑定

    在我们平时的开发中会经常用到Image控件,通过设置Image控件的Source属性,我们可以加载图片,设置Image的source属性时可以使用相对路径也可以使用绝对路径,一般情况下建议使用绝对路径 ...

  7. DAY07、字符编码和文件操作

    一.字符编码 1.什么是字符编码? 人类能识别的是字符等高级标识符,电脑只能识别0,1组成的标识符,要完成人与机器之间的信息交流,              一定需要一个媒介,进行两种标识符的转化(两 ...

  8. github---无命令可视化界面操作

    最近工作需要,研究了一下git,这个东西挺实用,给我的感觉并不是那么简单使用,我认为还可以再深入的研究一下,挺好玩的~ 说一下我的学习路线: 1.先看的廖老师的博客:https://www.liaox ...

  9. How to mount HFS EFI on macOS

    mount_hfs /dev/disk0s1 /volumes/efi

  10. CSS3之box-sizing属性

    box-sizing本身有三个属性:content-box(默认).border-box和padding-box. content-box:border与padding均不算入width中: bord ...