线段树
即使是最基础的线段树也有很多应用,比如什么优化dp啦,标记的神奇维护啦……
需要思路灵活一点,把题目条件抽象成更为简单的形式
扫描线
求矩形面积并,周长,二维数点等等
线段树作用即把静态O(n^2)变为动态O(nlogn).
想象过程,就是在一张图上,一条线从上到下扫描。所以线段树本质维护的是矩形的长,而宽又是从上到下排序过的,所以每次乘上差值即可。
常用手法比如把点的问题转化为矩形,可以求最大面积交/并
注意这个线段树写法和别的不太一样,有时需要离散化或动态开点。而且记录询问的数组开2倍。线段树必须开16倍空间,读入add就是二倍,线段树4倍,在pushup不可避免访问叶子节点的儿子,再有2倍。这里线段树是存的端点。
其实线段树可以直接存线段作为一个节点,这样在建树上需要考虑的东西更少,更不容易出错。
扫描线
#include<bits/stdc++.h>
#define ll long long
#define lid (id<<1)
#define rid (id<<1|1)
using namespace std;
const int maxn=100015;
struct node{ll x,yl,yr;int fl;
}p[maxn<<1];
struct tree{ll l,r,sum;
}t[maxn<<4];
int n,tg[maxn<<4];
ll s[maxn<<1],ans;
bool cmp(node a,node b){return a.x<b.x;
}
void pushup(int id){if(tg[id]>0) t[id].sum=t[id].r-t[id].l;//注意,这里代表是否矩形完全覆盖该位置 else t[id].sum=t[lid].sum+t[rid].sum;
}
void build(int id,int l,int r){t[id].l=s[l];//离散化,t[id]的l/r不是普遍意义 t[id].r=s[r];if(r-l==1){t[id].sum=0;return ;}int mid=(l+r)>>1;build(lid,l,mid);//维护的是面积,所以是按照矩形的边计算,mid相同 build(rid,mid,r);
//又,我这个离谱扫描线写法。。。其实也可以让线段树的每个点不是存节点,而是存一条边,它对于l,r,mid的处理就和普通线段树一样了pushup(id);
}
void add(int id,ll yl,ll yr,int fl){if(t[id].l==yl&&t[id].r==yr){tg[id]+=fl;pushup(id);return;}//这种写法实际上mid=t[lid].r=t[rid].l if(yl<t[lid].r) add(lid,yl,min(t[lid].r,yr),fl);if(yr>t[rid].l) add(rid,max(t[rid].l,yl),yr,fl);pushup(id);
}
int main(){scanf("%d",&n);for(int i=1;i<=n;i++){ll xl,yl,xr,yr;scanf("%lld%lld%lld%lld",&xl,&yl,&xr,&yr);p[i]=(node){xl,yl,yr,1};p[i+n]=(node){xr,yl,yr,-1};s[i]=yl,s[i+n]=yr;}sort(s+1,s+n*2+1);sort(p+1,p+n*2+1,cmp);build(1,1,2*n);add(1,p[1].yl,p[1].yr,p[1].fl);for(int i=2;i<=2*n;i++){ans+=(p[i].x-p[i-1].x)*t[1].sum;//printf("%d ",i);add(1,p[i].yl,p[i].yr,p[i].fl);}printf("%lld",ans);return 0;
}
主席树
可持久化权值线段树,经典应用是区间k大值
前置知识:权值线段树,动态开点线段树等
可以简单理解为对动态开点线段树做前缀和。新建的点连成的树和过去的树共用一部分节点。每次查询的时候用区间右减去区间左前一个的树,就得到了当前区间的树(和前缀和用法一样)。
主席树[luogu3834]
#include<bits/stdc++.h>
using namespace std;
const int maxn=2e5+5,mx=1e9;
int n,m,rt[maxn],cnt;
struct node{int lc,rc,siz;
}t[maxn<<5];
void add(int &x,int y,int l,int r,int val){x=++cnt,t[x]=t[y],t[x].siz++;if(l==r) return ;int mid=(l+r)>>1;if(val<=mid) add(t[x].lc,t[y].lc,l,mid,val);else add(t[x].rc,t[y].rc,mid+1,r,val);
}
int query(int x,int y,int l,int r,int k){if(l==r) return l;int mid=(l+r)>>1,tmp=t[t[y].lc].siz-t[t[x].lc].siz;if(tmp<k) return query(t[x].rc,t[y].rc,mid+1,r,k-tmp);else return query(t[x].lc,t[y].lc,l,mid,k);
}
int main(){scanf("%d%d",&n,&m);for(int i=1,x;i<=n;i++){scanf("%d",&x);add(rt[i],rt[i-1],0,mx,x);}while(m--){int l,r,k;scanf("%d%d%d",&l,&r,&k);printf("%d\n",query(rt[l-1],rt[r],0,mx,k));}return 0;
}
线段树优化建图
一看到点和区间或区间和区间连边的问题,大概率就是线段树优化建图了。
其基本思想就是自上而下连一个“出树”,自下而上连一个“入树”,对于其叶子节点(1-n)相互一一连边。然后按照题目要求从入树的节点/区间,连向出树的节点/区间。
一般建完图跑最短路的情况较多。
边数是(n+m)logn级别的,所以一般来说最短路跑出来是 \(O(nlog^2n)\) 的。
注意数组大小和空间限制!
[SNOI2017] 炸弹
//线段树优化建边+tarjan缩点+注意dp形式
//为什么re??
//有必要这样卡吗?
#include<bits/stdc++.h>
#define ll long long
#define lid (id<<1)
#define rid (id<<1|1)
using namespace std;
const ll mod=1e9+7;
const int maxn=5e5+5;
int cnt,h[maxn<<2],g[maxn<<2];
struct edge{int to,nxt;
}e[maxn*25],e1[maxn*25];
void add(int u,int v){e[++cnt].to=v;e[cnt].nxt=h[u];h[u]=cnt;
}
int cnt1,h1[maxn<<2],g1[maxn<<2];
void add1(int u,int v){e1[++cnt1].to=v;e1[cnt1].nxt=h1[u];h1[u]=cnt1;
}
int n,nx,k,num;
ll xx[maxn<<2],rr[maxn<<2],low[maxn<<2],dfn[maxn<<2],ins[maxn<<2],d[maxn<<2],la[maxn<<2],ra[maxn<<2];
struct node{ll l,r;
}t[maxn<<2];
void build(int id,int l,int r){t[id].l=l;t[id].r=r; nx=max(nx,id);if(l==r){g[l]=id;return ;}int mid=(l+r)>>1;build(lid,l,mid);build(rid,mid+1,r);add(id,lid);add(id,rid);
}
void xadd(int id,int lx,int rx,int l,int r,int q){if(l<=lx&&rx<=r){if(id!=q) add(q,id);//防止自环!!!! return ;}int mid=(lx+rx)>>1;if(l<=mid) xadd(lid,lx,mid,l,r,q);if(r>mid) xadd(rid,mid+1,rx,l,r,q);
}
stack<ll>s;
void tarjan(int x){low[x]=dfn[x]=++num;s.push(x);ins[x]=1;for(int i=h[x];i;i=e[i].nxt){int y=e[i].to;if(!dfn[y]){tarjan(y);low[x]=min(low[x],low[y]);}else if(ins[y]) low[x]=min(low[x],dfn[y]);}if(low[x]==dfn[x]){k++;int tmp=0;while(tmp!=x){tmp=s.top();s.pop();ins[tmp]=0;d[tmp]=k;la[k]=min(la[k],t[tmp].l);ra[k]=max(ra[k],t[tmp].r);}}
}
bool vis[maxn<<2];
void dfs(int x){vis[x]=1;for(int i=h1[x];i;i=e1[i].nxt){int y=e1[i].to;if(!vis[y]) dfs(y);la[x]=min(la[x],la[y]);ra[x]=max(ra[x],ra[y]);}
}
int main(){scanf("%d",&n);for(int i=1;i<=n;i++){scanf("%lld%lld",&xx[i],&rr[i]);}build(1,1,n);xx[n+1]=0x3f3f3f3f3f3f3f3f;//便于查找位置 for(int i=1;i<=n;i++){if(!rr[i]) continue;int tl=lower_bound(xx+1,xx+n+1,xx[i]-rr[i])-xx;int tr=upper_bound(xx+1,xx+n+1,xx[i]+rr[i])-xx-1;xadd(1,1,n,tl,tr,g[i]);}memset(la,0x3f,sizeof(la));tarjan(1);for(int i=1;i<=nx;i++){//注意这里是build后的 for(int j=h[i];j;j=e[j].nxt){int y=e[j].to;if(d[i]!=d[y]){add1(d[i],d[y]);}}}for(int i=1;i<=k;i++){if(!vis[i]){dfs(i);}}ll ans=0;for(int i=1;i<=n;i++){ans=(ans+(ll)i*(ra[d[g[i]]]-la[d[g[i]]]+1)%mod)%mod;}printf("%lld",ans);return 0;
}
线段树合并
新建/不新建节点
还可以回收节点!
线段树分裂
有点像FHQ-treap的 split函数(分裂?
比如权值线段树以k为界分裂 split(x,y,k),即分裂x,给y
val[lc[x]]<k,左子树不用修改,直接 split(rc[x],rc[y],k-v)
val[lc[x]]=k,左子树正好包含前k个,于是将右子树给y,rc[y]=rc[x],rc[x]=0
val[lc[x]]>k,右子树全大于k,直接给y,递归左子树,split(lc[x],lc[y],k)
每次只会递归一边,单次时间复杂度O(logn)
线段树分裂/合并板子
#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn=2e5+5;
int n,m,cnt,rt[maxn<<2],crt=1;
ll a[maxn];
struct node{int lc,rc; ll v;
}t[maxn<<5];
ll read(){char ch=getchar(); ll x=0;while(ch<'0'||ch>'9') ch=getchar();while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar();return x;
}
void write(ll x){if(x>9) write(x/10);putchar('0'+x%10);
}
//int newnode(){
// return dcnt?p[dcnt--]:++cnt;//关于回收点,不写也是对的
//}
//void del(int x){
// p[++dcnt]=x;
// t[x].lc=t[x].rc=t[x].v=0;
//}
void add(int &x,int l,int r,int pos,ll val){if(!x) x=++cnt;if(l==r){t[x].v+=val; return ;}int mid=(l+r)>>1;if(pos<=mid) add(t[x].lc,l,mid,pos,val);else add(t[x].rc,mid+1,r,pos,val);t[x].v=t[t[x].lc].v+t[t[x].rc].v;
}
ll query(int x,int l,int r,int vl,int vr){ if(!x||vl>r||vr<l) return 0;if(vl<=l&&r<=vr) return t[x].v;int mid=(l+r)>>1; ll res=0;if(vl<=mid) res=query(t[x].lc,l,mid,vl,vr);if(vr>mid) res+=query(t[x].rc,mid+1,r,vl,vr);return res;
}
int kth(int x,int l,int r,ll val){if(l==r) return l;int mid=(l+r)>>1;if(t[t[x].lc].v>=val) return kth(t[x].lc,l,mid,val);else return kth(t[x].rc,mid+1,r,val-t[t[x].lc].v);
}
int merge(int x,int y){if(!x||!y) return x+y;t[x].v+=t[y].v;t[x].lc=merge(t[x].lc,t[y].lc);t[x].rc=merge(t[x].rc,t[y].rc);
// del(y);return x;
}
void split(int x,int &y,ll k){if(!x) return ;y=++cnt;ll val=t[t[x].lc].v;if(k>val) split(t[x].rc,t[y].rc,k-val);else swap(t[x].rc,t[y].rc);//k<=valif(k<val) split(t[x].lc,t[y].lc,k);t[y].v=t[x].v-k,t[x].v=k;//存的是个数
}
signed main(){n=read(),m=read(); for(int i=1;i<=n;i++){a[i]=read();add(rt[1],1,n,i,a[i]);}for(int i=1;i<=m;i++){int opt=read(),x=read(),y=read(); ll z;if(opt==0){z=read();ll q1=query(rt[x],1,n,1,z),q2=query(rt[x],1,n,y,z);int tmp=0;crt++,split(rt[x],rt[crt],q1-q2),split(rt[crt],tmp,q2);//注意split还是要先跑出数量来分裂,主要是可能没有等于k的点,难以处理分裂 rt[x]=merge(rt[x],tmp);}else if(opt==1){rt[x]=merge(rt[x],rt[y]);}else if(opt==2){z=read();add(rt[x],1,n,z,y);}else if(opt==3){z=read(); write(query(rt[x],1,n,y,z)),putchar('\n');} else{if(t[rt[x]].v<1ll*y) printf("-1\n");else write(kth(rt[x],1,n,y)),putchar('\n');}}return 0;
}
线段树分治
离线算法,我们对时间(询问)建立一颗线段树,树的节点存的信息是“覆盖了这个区间的操作”。一般是 \(O(nlog^2n)\) 左右的。
实现上每遍历到一个节点就去add,出了某个节点需要del,用一个可撤销并查集维护。
Q: 为什么要建线段树呢?遍历到l的时候加入,到r的时候删除不好吗?
A:可撤销并查集必须保证插入和删除的顺序完全相同,是一个栈。如果不建线段树根本无法使用可撤销并查集。
注意,可撤销并查集只能按秩合并/启发式合并,不能路径压缩!(按秩合并更快,但不好写
使用情景:每个操作影响范围是一个区间,多组操作后询问。(还经常实用扩展域并查集!)
Pastoral Oddities
//非常非常巧妙,首先发现性质,只有偶数连通块
//其次注意到kruskal单调不增的性质,某个边一旦被淘汰就不会再被选中
//还是从小到大去做,注意从后向前去选择,因为答案单调
//半在线线段树分治
#include<bits/stdc++.h>
#define lid (id<<1)
#define rid (id<<1|1)
using namespace std;
const int maxn=5e5+5;
int n,m,k,q,ans[maxn],s[maxn<<3],top,cnt,num;
struct node{int x,y,id,z;
}e[maxn<<1];
int f[maxn<<1],siz[maxn<<1];
bool cmp(node a,node b){return a.z<b.z;
}
int find(int x){return (f[x]==x)?x:find(f[x]);
}
void merge(int x,int y){x=find(x),y=find(y);if(x==y) return ;if(siz[x]>siz[y]) swap(x,y);cnt-=(siz[x]&1)+(siz[y]&1);siz[y]+=siz[x],f[x]=y;s[++top]=x;cnt+=(siz[y]&1);
}
void del(int now){while(top>now){int x=s[top--],y=f[x];cnt-=(siz[y]&1);siz[y]-=siz[x];cnt+=(siz[x]&1)+(siz[y]&1);f[x]=x; }
}
vector<int>v[maxn<<3];
void upd(int id,int l,int r,int vl,int vr,int pos){if(vl>vr) return ;if(vl<=l&&r<=vr){v[id].push_back(pos);return ;}int mid=(l+r)>>1;if(vl<=mid) upd(lid,l,mid,vl,vr,pos);if(vr>mid) upd(rid,mid+1,r,vl,vr,pos);
}
void sol(int id,int l,int r){int ttop=top;for(auto i:v[id]){merge(e[i].x,e[i].y);}if(l==r){while(cnt&&num<m){num++;if(e[num].id<=l){upd(1,1,m,e[num].id,l-1,num);merge(e[num].x,e[num].y);}}if(cnt==0) ans[l]=e[num].z;else ans[l]=-1;del(ttop);return ;}int mid=(l+r)>>1;sol(rid,mid+1,r),sol(lid,l,mid);del(ttop);
}
int read(){char ch=getchar(); int x=0;while(ch<'0'||ch>'9') ch=getchar();while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar();return x;
}
int main(){n=read(),m=read();for(int i=1;i<=m;i++){e[i].x=read(),e[i].y=read(),e[i].z=read();e[i].id=i;}for(int i=1;i<=n;i++) f[i]=i,siz[i]=1;cnt=n;sort(e+1,e+m+1,cmp);sol(1,1,m);for(int i=1;i<=m;i++){printf("%d\n",ans[i]);}return 0;
}
线段树二分
简而言之就是直接在线段树上跑二分,就和单点插入时那个寻找mid和pos关系比较像(包括类似于就是权值线段树查询k小值的写法
if(true) return sol(rid,mid+1,r,p);
else return sol(lid,l,mid,p);
优点是不用单独写二分,少一个log的复杂度,即总复杂度为O((n+q)logn)
树套树
主席树本来是不支持修改的。
但考虑,主席树其实就是对很多树链做了前缀和。那么对于单点修改(要求的修改操作),区间查询(原本主席树支持的),不就是用树状数组维护吗?!
所以就诞生了树套树中树状数组套主席树的写法。(当然你也可以套平衡树)
至于复杂度,树状数组的单点修改区间查询都是logn的,那么总时间复杂度 \(O(nlog^2n)\)
树套树
//树状数组套主席树
#include<bits/stdc++.h>
using namespace std;
const int maxn=5e4+5,mx=1e8;
int n,m,p1[maxn<<5],p2[maxn<<5],tp1[maxn<<5],tp2[maxn<<5],num1,num2;
int rt[maxn<<2],a[maxn<<2];
inline int lowbit(int x){return x&(-x);
}
inline int read(){char ch=getchar(); int x=0;while(ch<'0'||ch>'9') ch=getchar();while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar();return x;
}
inline void write(int x){if(x<0) putchar('-'),x=-x;if(x>9) write(x/10);putchar('0'+x%10);
}
struct seg{int cnt=0;struct node{int v,lc,rc;}t[maxn<<10];inline void add(int &x,int l,int r,int pos,int val){if(!x) x=++cnt; t[x].v+=val;if(l==r) return;int mid=(l+r)>>1;if(pos<=mid) add(t[x].lc,l,mid,pos,val);else add(t[x].rc,mid+1,r,pos,val);}//p2-p1inline int query(int p1[],int p2[],int l,int r,int pos){if(l==r) return 0;//可爱的边界条件……这里注意,为了方便后面查询前后继,不要算上等于 int mid=(l+r)>>1;if(pos<=mid){for(int i=1;i<=num1;i++) p1[i]=t[p1[i]].lc;for(int i=1;i<=num2;i++) p2[i]=t[p2[i]].lc;return query(p1,p2,l,mid,pos);} else{int tmp=0;for(int i=1;i<=num1;i++) tmp-=t[t[p1[i]].lc].v,p1[i]=t[p1[i]].rc;for(int i=1;i<=num2;i++) tmp+=t[t[p2[i]].lc].v,p2[i]=t[p2[i]].rc;return tmp+query(p1,p2,mid+1,r,pos);}}inline int kth(int p1[],int p2[],int l,int r,int k){if(l==r) return l;int tmp=0;for(int i=1;i<=num1;i++) tmp-=t[t[p1[i]].lc].v;for(int i=1;i<=num2;i++) tmp+=t[t[p2[i]].lc].v;int mid=(l+r)>>1;if(k>tmp){for(int i=1;i<=num1;i++) p1[i]=t[p1[i]].rc;for(int i=1;i<=num2;i++) p2[i]=t[p2[i]].rc;return kth(p1,p2,mid+1,r,k-tmp);} else{for(int i=1;i<=num1;i++) p1[i]=t[p1[i]].lc;for(int i=1;i<=num2;i++) p2[i]=t[p2[i]].lc;return kth(p1,p2,l,mid,k);}}
}T;
inline void add(int x,int val){int i=x;while(x<=n){T.add(rt[x],0,mx,a[i],-1);x+=lowbit(x);}a[i]=val,x=i;while(x<=n){T.add(rt[x],0,mx,a[i],1);x+=lowbit(x);}
}
inline void query(int x,int y,int k,int fl){num1=num2=0,x--;int len=y-x;while(x){num1++,p1[num1]=tp1[num1]=rt[x];x-=lowbit(x);}while(y){num2++,p2[num2]=tp2[num2]=rt[y];y-=lowbit(y);}if(fl==1) write(T.query(p1,p2,0,mx,k)+1);else if(fl==2) write(T.kth(p1,p2,0,mx,k));else{if(fl==4){if(k==0) write(-2147483647);else{x=T.query(p1,p2,0,mx,k);write((x==0)?-2147483647:T.kth(tp1,tp2,0,mx,x)); }}else{if(k==mx) write(2147483647);else{x=T.query(p1,p2,0,mx,k+1);write((x==len)?2147483647:T.kth(tp1,tp2,0,mx,x+1));}}}putchar('\n');
}
int main(){
// system("fc 1.out ans.out");
// freopen("1.in","r",stdin);
// freopen("ans.out","w",stdout);n=read(),m=read();for(int i=1;i<=n;i++){a[i]=read();for(int j=i;j<=n;j+=lowbit(j))T.add(rt[j],0,mx,a[i],1);//这个add好像必须这样写,因为要新建点! }while(m--){int opt=read(),l=read(),r=read(),k;if(opt==3) add(l,r);else{k=read();query(l,r,k,opt);}}return 0;
}
吉司机线段树 Seg-beats
经典问题是,给定一个序列,支持区间取min/max和区间求和。时间复杂度O((n+q)logn)
以维护区间取min为例,线段树每个节点维护mx(区间最大值),cnt(区间最大值的出现次数),cmx(区间严格次大值),sum(区间和).
做区间取min时,递归到某个节点p
- \(x \geq mx_p\) ,则这次修改不影响节点p,直接return
2)\(x \leq cmx_p\),则直接暴力往p的左右子节点递归
3)\(cmx_p < x < mx_p\),则这次修改对 \(sum_p\) 的贡献为 \(-cnt_p*(mx_p-x)\),\(mx_p=x\)
复杂度分析:需要用到势能分析。
设线段树T上,“一类点”为其最大值等于父亲的最大值的点,二类点为其最大值小于父亲的最大值的点。
设一棵线段树的势能函数\(Φ(T)\)为二类点的数量,则\(Φ(T) \leq n\)
每次区间取min的操作更新下来,会有若干二类点变成一类点,而变一个二类点的花费是O(logn)的(一条路径顺下来),一类点受到更改是O(1)的
那么均摊下来,总复杂度不超过\(O(nlogn)\)
注意,吉司机线段树也可以维护区间加,多带一个加法标记即可,但复杂度要多一个log,复杂度分析类似
历史最值线段树
实际上就是多维护了一个标记,hadd表示历史上最大的addtag,每次先下传历史最大add再下传覆盖/修改等标记。
[模板]线段树3(吉司机线段树+历史最值)
#include<bits/stdc++.h>
#define ll long long
#define lid (id<<1)
#define rid (id<<1|1)
using namespace std;
const int maxn=5e5+5;
const ll inf=-(1ll<<61);
int n,m;
ll a[maxn];
struct node{int l,r,cnt;ll ad,had,cad,hcad;//hadd没更新之前的最大tag,最大值和次大值分开维护 ll mx,cmx,hmx,sum;
}t[maxn<<2];
void upd(int id,ll ad,ll had,ll cad,ll hcad){t[id].sum+=ad*t[id].cnt+cad*(t[id].r-t[id].l+1-t[id].cnt);t[id].hmx=max(t[id].hmx,t[id].mx+had);//注意上就是先下传之前的maxadd t[id].mx+=ad;if(t[id].cmx!=inf) t[id].cmx+=cad;t[id].had=max(t[id].had,t[id].ad+had);t[id].hcad=max(t[id].hcad,t[id].cad+hcad);t[id].ad+=ad,t[id].cad+=cad;
}
void pushdown(int id){ll tmp=max(t[lid].mx,t[rid].mx);//傻了。。这才是要pushdown去更新的max,因为当前节点的max已经被更改,所以只看下面的最大值 if(tmp==t[lid].mx) upd(lid,t[id].ad,t[id].had,t[id].cad,t[id].hcad);else upd(lid,t[id].cad,t[id].hcad,t[id].cad,t[id].hcad);if(tmp==t[rid].mx) upd(rid,t[id].ad,t[id].had,t[id].cad,t[id].hcad);else upd(rid,t[id].cad,t[id].hcad,t[id].cad,t[id].hcad);t[id].ad=t[id].had=t[id].cad=t[id].hcad=0;
}
void pushup(int id){if(t[lid].mx>t[rid].mx){t[id].mx=t[lid].mx,t[id].cnt=t[lid].cnt;t[id].cmx=max(t[lid].cmx,t[rid].mx);}else if(t[lid].mx<t[rid].mx){t[id].mx=t[rid].mx,t[id].cnt=t[rid].cnt;t[id].cmx=max(t[lid].mx,t[rid].cmx);}else{t[id].mx=t[lid].mx,t[id].cnt=t[lid].cnt+t[rid].cnt;t[id].cmx=max(t[lid].cmx,t[rid].cmx);}t[id].hmx=max(t[lid].hmx,t[rid].hmx);t[id].sum=t[lid].sum+t[rid].sum;
}
void build(int id,int l,int r){t[id].l=l,t[id].r=r;if(l==r){t[id].mx=t[id].hmx=t[id].sum=a[l];t[id].cnt=1,t[id].cmx=inf;return ;}int mid=(l+r)>>1;build(lid,l,mid),build(rid,mid+1,r);pushup(id);
}
void add(int id,int l,int r,int vl,int vr,ll val){if(vl<=l&&r<=vr){upd(id,val,val,val,val);return ;}int mid=(l+r)>>1; pushdown(id);if(vl<=mid) add(lid,l,mid,vl,vr,val);if(vr>mid) add(rid,mid+1,r,vl,vr,val);pushup(id);
}
void change(int id,int l,int r,int vl,int vr,ll val){if(val>=t[id].mx) return ;if(vl<=l&&r<=vr&&t[id].cmx<val){t[id].sum-=1ll*t[id].cnt*(t[id].mx-val);t[id].ad-=t[id].mx-val,t[id].mx=val;return ;}int mid=(l+r)>>1; pushdown(id);if(vl<=mid) change(lid,l,mid,vl,vr,val);if(vr>mid) change(rid,mid+1,r,vl,vr,val);pushup(id);
}
ll query(int id,int l,int r,int vl,int vr){if(vl<=l&&r<=vr) return t[id].sum;int mid=(l+r)>>1; ll res=0; pushdown(id);if(vl<=mid) res=query(lid,l,mid,vl,vr);if(vr>mid) res+=query(rid,mid+1,r,vl,vr);return res;
}
ll querym(int id,int l,int r,int vl,int vr){if(vl<=l&&r<=vr) return t[id].mx;int mid=(l+r)>>1; ll res=inf;pushdown(id);if(vl<=mid) res=querym(lid,l,mid,vl,vr);if(vr>mid) res=max(res,querym(rid,mid+1,r,vl,vr));return res;
}
ll queryh(int id,int l,int r,int vl,int vr){if(vl<=l&&r<=vr) return t[id].hmx;int mid=(l+r)>>1; ll res=inf;pushdown(id);if(vl<=mid) res=queryh(lid,l,mid,vl,vr);if(vr>mid) res=max(res,queryh(rid,mid+1,r,vl,vr));return res;
}
int main(){scanf("%d%d",&n,&m);for(int i=1;i<=n;i++){scanf("%lld",&a[i]);}build(1,1,n);for(int i=1;i<=m;i++){int op,l,r; ll k;scanf("%d%d%d",&op,&l,&r);if(op==1) scanf("%lld",&k),add(1,1,n,l,r,k);else if(op==2) scanf("%lld",&k),change(1,1,n,l,r,k);else if(op==3) printf("%lld\n",query(1,1,n,l,r));else if(op==4) printf("%lld\n",querym(1,1,n,l,r));else printf("%lld\n",queryh(1,1,n,l,r));}return 0;
}
李超线段树
最经典应用就是维护一个二维平面直角坐标系,支持动态插入线段(理解为有值域的一次函数 \(y=kx+b\) ),询问已插入线段与直线 \(x=x_0\) 交点的纵坐标最值。
即当 \(x=x_0\) ,求 \(max\) 或 \(min\) { \(k_ix+b_i\) }
对于普通求法的 \(O(n)\) 遍历,如何优化时间复杂度呢?其实就是找一种方法减少有效集合大小,而李超线段树就是如此。
单次查询是 \(O(logn)\) 的,修改是 \(O(log^2n)\) 的。
现在我们需要插入一条线段 f,在这条线段完整覆盖的线段树节点代表的区间中,某些区间的最优线段可能发生改变。
考虑某个被新线段 f 完整覆盖的区间,若该区间无最优线段,则该线段可以直接成为最优线段。
否则,设该区间的中点为 mid,我们拿新线段 f 在中点处的值与原最优线段 g 在中点处的值作比较。
如果新线段 f 更优,则将 f 和 g 交换。
那么现在考虑在中点处 f 不如 g 优的情况:
若在左端点处 f 更优,那么f 和 g 必然在左半区间中产生了交点,递归到左儿子中进行插入;
若在右端点处 f 更优,那么 f 和 g 必然在右半区间中产生了交点,递归到右儿子中进行插入。
若在左右端点处 g 都更优,那么 f 不可能成为答案,不需要继续下传。
这个方法的优势就在于不需要讨论斜率的正负,十分方便。
李超线段树[模板] [HEOI2013]Segment
#include<bits/stdc++.h>
#define lid (id<<1)
#define rid (id<<1|1)
#define pdi pair<double,int>
using namespace std;
const double eps=1e-9;//精度问题
const int maxn=1e5+5,mod1=39989,mod2=1000000000;
int n,cnt,ans,t[maxn<<1];
struct node{double k,b;
}p[maxn];
int cmp(double x,double y){if(x-y>eps) return 1;if(y-x>eps) return -1;return 0;
}
double f(int id,int x){return p[id].b+p[id].k*x;
}
void add(int x,int y,int xx,int yy){cnt++;if(x==xx) p[cnt].k=0,p[cnt].b=max(y,yy);//特判直线斜率不存在的情况!else p[cnt].k=1.0*(yy-y)/(xx-x),p[cnt].b=y-p[cnt].k*x;
}
void upd(int id,int l,int r,int u){//对线段完全覆盖到的区间进行修改 int &v=t[id],mid=(l+r)>>1;int tmp=cmp(f(u,mid),f(v,mid));if(tmp==1||(!tmp&&u<v)) swap(u,v);//交换新旧线段,这里是取地址的 int tl=cmp(f(u,l),f(v,l)),tr=cmp(f(u,r),f(v,r));if(tl==1||(!tl&&u<v)) upd(lid,l,mid,u);if(tr==1||(!tr&&u<v)) upd(rid,mid+1,r,u);
}
void change(int id,int l,int r,int vl,int vr,int u){if(vl<=l&&r<=vr){upd(id,l,r,u);return ;}int mid=(l+r)>>1;//线段树的查询 if(vl<=mid) change(lid,l,mid,vl,vr,u);if(mid<vr) change(rid,mid+1,r,vl,vr,u);
}
pdi pmax(pdi x,pdi y){if(cmp(x.first,y.first)==-1) return y;else if(cmp(x.first,y.first)==1) return x;else return x.second<y.second?x:y;
}
pdi query(int id,int l,int r,int pos){if(r<pos||pos<l) return {0,0};//一种比较方便的写法 double res=f(t[id],pos);if(l==r) return {res,t[id]};int mid=(l+r)>>1;return pmax({res,t[id]},pmax(query(lid,l,mid,pos),query(rid,mid+1,r,pos)));
}
int main(){scanf("%d",&n);while(n--){int op;scanf("%d",&op);if(op==1){int x,y,xx,yy;scanf("%d%d%d%d",&x,&y,&xx,&yy);x=(x+ans-1+mod1)%mod1+1;xx=(xx+ans-1+mod1)%mod1+1;y=(y+ans-1+mod2)%mod2+1;yy=(yy+ans-1+mod2)%mod2+1;if(x>xx) swap(x,xx),swap(y,yy);add(x,y,xx,yy);change(1,1,mod1,x,xx,cnt);}else{int x;scanf("%d",&x);x=(x+ans-1+mod1)%mod1+1;ans=query(1,1,mod1,x).second;printf("%d\n",ans);}}return 0;
}
珂朵莉树
珂朵莉树(老司机树),一种暴力数据结构,做一些区间修改区间查询的问题,对随机数据比较有效
对一个序列,进行一个区间推平操作(如把一个区间内变为同一个数),然后我们把每段插入到set中自动排序
对于其他操作,还是比较暴力的(对于每个段进行操作)……复杂度在随机数据下大概是O(nlog^2n)
柯朵莉树/老司机树(odt)
#include<bits/stdc++.h>
#define int long long
using namespace std;
const int mod=1e9+7,maxn=1e5+5;
int n,m,seed,vmax,a[maxn];
struct node{int l,r;mutable int v;//这一段相同元素的值,即使是常量也可以被修改bool operator<(const node &a) const{return l<a.l;}
};
set<node>s;
set<node>::iterator split(int pos){//分裂成[l,pos-1] [pos,r] set<node>::iterator it=s.lower_bound((node){pos,0,0});if(it!=s.end()&&it->l==pos) return it;it--;if(it->r<pos) return s.end();int l=it->l,r=it->r,v=it->v;s.erase(it);s.insert((node){l,pos-1,v});return s.insert((node){pos,r,v}).first;
}
void assign(int l,int r,int x){set<node>::iterator itr=split(r+1),itl=split(l);//注意顺序,否则会因操作r时删去了l的指针而re s.erase(itl,itr);//前闭后开 s.insert((node){l,r,x});
}
void add(int l,int r,int x){set<node>::iterator itr=split(r+1),itl=split(l);for(auto it=itl;it!=itr;it++){it->v+=x;//it是常量迭代器,本来不能下哦i高,但是加了mutable就好了 }
}
struct rankk{int num,cnt;bool operator<(const rankk &a)const{return num<a.num;}
};
int rnk(int l,int r,int x){set<node>::iterator itr=split(r+1),itl=split(l);vector<rankk>v;for(auto i=itl;i!=itr;i++){v.push_back((rankk){i->v,i->r-i->l+1});}sort(v.begin(),v.end());int i;for(i=0;i<v.size();i++){if(v[i].cnt<x) x-=v[i].cnt;else break;}return v[i].num;
}
int qpow(int x,int y,int mod){int res=1; x%=mod;while(y){if(y&1) res=res*x%mod;x=x*x%mod;y>>=1;}return res;
}
int query(int l,int r,int x,int y){set<node>::iterator itr=split(r+1),itl=split(l);int ans=0;for(auto i=itl;i!=itr;i++){ans=(ans+qpow(i->v,x,y)*(i->r-i->l+1)%y)%y;}return ans;
}
int rnd(){int ret=seed;seed=(seed*7+13)%mod;return ret;
}
signed main(){scanf("%lld%lld%lld%lld",&n,&m,&seed,&vmax);for(int i=1;i<=n;i++){a[i]=rnd()%vmax+1;s.insert((node){i,i,a[i]});}for(int i=1;i<=m;i++){int opt,l,r,x,y;opt=rnd()%4+1;l=rnd()%n+1;r=rnd()%n+1;if(l>r) swap(l,r);if(opt==3) x=rnd()%(r-l+1)+1;else x=rnd()%vmax+1;if(opt==4) y=rnd()%vmax+1,printf("%lld\n",query(l,r,x,y));else if(opt==3) printf("%lld\n",rnk(l,r,x));else if(opt==2) assign(l,r,x);else add(l,r,x);}return 0;
}