线段树是一种能够有效处理区间操作的高级数据结构,利用这种数据结构,我们能够设计出针对上述问题更加高效的算法。
线段树的题目通常比较明显,一般一个很明显的特征是m次对某一区间长度的查询。或者是修改。所以我们通常需要的只是将线段树的模型稍加修改,进而套入题目中即可。
模板:
对单个点的修改,和对一段区间的查询:
//线段树单个点修改&区间查询 #include<iostream> #include<iomanip> #include<cstring> #include<climits> #include<cmath> #include<cstdio> #include<cstdlib> #include<queue> #include<vector> #include<map> #include<algorithm> #include<string> #include<memory> using namespace std; const int e=100006; struct qq { int maxx; }tree[4*e];//线段树要开4倍的点的个数 int n,t,a,b; void updata(int l,int r,int root) { if(r<a || l>a) return; if(r==l) { tree[root].maxx=b; return; } int mid=(l+r)/2; updata(l,mid,root*2); updata(mid+1,r,root*2+1); tree[root].maxx=max(tree[root*2].maxx,tree[root*2+1].maxx); } int search(int l,int r,int root) { if(l>b || r<a) return(-999999999); if(l>=a && r<=b) return(tree[root].maxx); int mid=(l+r)/2; return(max( search(l,mid,root*2), search(mid+1,r,root*2+1))); } int main() { memset(tree,0,sizeof(tree)); cin>>n; for(int i=0;i<n;i++) { scanf("%d%d%d",&t,&a,&b); if(t==1) updata(1,n,1);//将点a的值改为b if(t==2) cout<< search(1,n,1) << endl;//查找区间a(含)到b(含)的最大值; } return 0; }
对一段区间的修改和查询:
//线段树 区间 修改&查询 #include<iostream> #include<iomanip> #include<cstring> #include<climits> #include<cmath> #include<cstdio> #include<cstdlib> #include<queue> #include<vector> #include<map> #include<algorithm> #include<string> #include<memory> using namespace std; const int e=100006; struct qq { int maxx,delta; }tree[4*e];//线段树要开4倍的点的个数 int n,t,a,b; void updata(int l,int r,int root)//更新数据 { if(b<l || r<a) return; if(l>=a && r<=b)//这句是核心,若当前区间包含于修改区间,就不往下传,(未传到叶子节点); { tree[root].maxx++; tree[root].delta++; return; } int mid=(l+r)/2,delta=tree[root].delta; tree[root*2].maxx+=delta; tree[root*2].delta+=delta; tree[root*2+1].maxx+=delta; tree[root*2+1].delta+=delta; tree[root].delta=0;//这句很关键,根的偏移量传递到子树后清零 updata(l,mid,root*2); updata(mid+1,r,root*2+1); tree[root].maxx=max( tree[root*2].maxx, tree[root*2+1].maxx); return; } int search(int l,int r,int root) { if(l>b || r<a) return(-99999999); if(l>=a && r<=b) return(tree[root].maxx); int mid=(l+r)/2,delta=tree[root].delta; tree[root*2].maxx+=delta; tree[root*2].delta+=delta; tree[root*2+1].maxx+=delta; tree[root*2+1].delta+=delta; tree[root].delta=0; //这句很关键,根的偏移量传递到子树后清零 return(max( search(l,mid,root*2), search(mid+1,r,root*2+1))); } int main() { memset(tree,0,sizeof(tree)); cin>>n; for(int i=0;i<n;i++) { scanf("%d%d%d",&t,&a,&b); if(t==1) updata(1,n,1); if(t==2) cout<< search(1,n,1) << endl; } return 0; }以下是做线段树题目时的易错点:
1、开线段树的结构体时一定要开到4倍的点的大小;
2、写区间求和时的题目时search()函数的返回值在不再区间里时返回0,而区间求最小值时返999999999,最大值时返回-999999999;(int时)一定要是九个9,不然有些极限数据会卡范围;
3、写修改单个点的值和一个区间的值的函数最好分开,一个是节省时间,还有可以防止代码混乱而出错;
4、Search()&updata()函数中判断区间范围的if语句中l,r和查找的区间a,b的关系容易出错;
5、注意根节点和子节点的关系,特别是+1的问题,归结为一句话就是:若加都加,若不加都不加即:mid=(l+r);左子树:l~mid(不加一),根为root*2;右子树:r~mid+1(加一了),根为root*2+1(也加一);
6、权值是在边上还是点上,这两种关系代码判断上有不同;
7、结构体赋初值的时候也要注意,根据所求的是最大还是最小还是和来判断;
8、线段树的左端点在数组中的下标一定要是1,而不是0。