對於長度爲$n$的初始正整數數列$A$,定義矩陣$B_{n×n}$,初始時$B_{i,j}=\sum\limits_{k=i}^{j}A_k$, 。node
有$m$個操做,每一個操做爲如下兩種之一:c++
- 給出整數$p, x$,將$A_p$修改成$x$。而後對全部的$B_{i,j}$,更新爲$min(B_{i,j},\sum\limits_{k=i}^{j} A_k)$。git
- 給出整數$l,r$,請回答$B_{l,r}$的值。算法
$KDtree$ 的基本思路是對於二維的問題,交替在一維上進行分治,嚴格來講效率是和主席樹相似的,可是因爲特殊的切割方式對於一個區間覆蓋到的塊更少,同時空間效率也更優秀,是 $O(n)$ 的。網上說這個東西效率是 $O(n\sqrt{n})$ 的,那它就是把...ide
#include<bits/stdc++.h> #define il inline #define LL long long #define _(d) while(d(isdigit(ch=getchar()))) using namespace std; const int N=1e5+5; LL sum[N]; int a[N],o,pos[N],rt,tot,idx[N],n,m; struct node{ int op,x,y; }f[N]; struct data{ int id,d[2],mn[2],mx[2],fa,ls,rs; LL tag,tm,now,s; bool operator<(const data&t1)const{ return d[o]<t1.d[o]; } }t[N]; il int read(){ int x,f=1;char ch; _(!)ch=='-'?f=-1:f;x=ch^48; _()x=(x<<1)+(x<<3)+(ch^48); return f*x; } il void add(int i,int l,int r){ t[i].id=i;t[i].d[0]=l;t[i].d[1]=r; } il void update(int fa,int x){ t[x].fa=fa; for(int i=0;i<2;i++) t[fa].mx[i]=max(t[fa].mx[i],t[x].mx[i]),t[fa].mn[i]=min(t[fa].mn[i],t[x].mn[i]); } il int build(int l,int r,int op){ int m=(l+r)>>1; o=op;nth_element(t+l,t+m,t+r+1); t[m].mn[0]=t[m].mx[0]=t[m].d[0]; t[m].mn[1]=t[m].mx[1]=t[m].d[1]; if(l<m)t[m].ls=build(l,m-1,op^1),update(m,t[m].ls); if(m<r)t[m].rs=build(m+1,r,op^1),update(m,t[m].rs); return m; } il void W(int x,int y){ t[x].s=min(t[x].s,t[x].now+t[y].tm);t[x].now+=t[y].tag; t[x].tm=min(t[x].tm,t[x].tag+t[y].tm);t[x].tag+=t[y].tag; } il void pushdown(int x){ if(t[x].tag==0&&t[x].tm==0)return; if(t[x].ls)W(t[x].ls,x);if(t[x].rs)W(t[x].rs,x); int kk=t[x].ls; t[x].tag=t[x].tm=0; } il void pushup(int x){ if(t[x].fa>0)pushup(t[x].fa);pushdown(x); } il void Add(int x,int p,int v){ if(!x||t[x].mn[0]>p||t[x].mx[1]<p)return; if(t[x].mx[0]<=p&&t[x].mn[1]>=p){ t[x].tag+=v;t[x].tm=min(t[x].tm,t[x].tag); t[x].now+=v;t[x].s=min(t[x].s,t[x].now); return; } if(t[x].d[0]<=p&&p<=t[x].d[1])t[x].now+=v,t[x].s=min(t[x].s,t[x].now); pushdown(x);Add(t[x].ls,p,v);Add(t[x].rs,p,v); } il LL query(int x){ pushup(x);return t[x].s; } int main() { n=read();m=read(); for(int i=1;i<=n;i++)a[i]=read(),sum[i]=sum[i-1]+a[i]; for(int i=1;i<=m;i++){ int op=read(),x=read(),y=read(); if(op==1){ int v=y-a[x];a[x]=y;f[i]=(node){op,x,v}; } else{ f[i]=(node){op,x,y};add(++tot,x,y);idx[tot]=i; } } rt=build(1,tot,0); for(int i=1;i<=tot;i++)pos[idx[t[i].id]]=i; for(int i=1;i<=m;i++){ if(f[i].op==1)Add(rt,f[i].x,f[i].y); else printf("%lld\n",sum[f[i].y]-sum[f[i].x-1]+query(pos[i])); } return 0; }