-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSegmentTree.c
90 lines (74 loc) · 2.92 KB
/
SegmentTree.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
#include <stdio.h>
#include<limits.h>
#include<math.h>
int min(int a,int b){ //for returning miniumum
return((a<b)? a:b);
}
void build(int a[],int seg[],int low, int high,int pos){ //build segment tree
if(high==low){seg[pos]=a[low]; return;} // condition to check leaf node
int mid=(high+low)/2;
build(a,seg,low,mid,pos*2+1); //left child
build(a,seg,mid+1,high,pos*2+2); //right child
seg[pos]=min(seg[2*pos+1],seg[2*pos+2]); //setting parent node according to the requirement
}
int query(int seg[],int lazy[],int low, int high,int qlow, int qhigh,int pos ){ // query function
if(high<low)return INT_MAX;
if(lazy[pos]!=0){
seg[pos]+=lazy[pos]; //upating the segment tree according to lazy tree
if(high!=low){ // if this is not the leaf node then propagate the value
lazy[2*pos+1]+=lazy[pos]; // to left child node
lazy[2*pos+2]+=lazy[pos]; // to right child node
}
lazy[pos]=0;
}
if(qlow<=low&&qhigh>=high)return seg[pos]; // total overlapping
if(qlow>high||qhigh<low)return INT_MAX; // no overlapping
int mid=(high+low)/2;
return min(query(seg,lazy,low,mid,qlow, qhigh,pos*2+1),query(seg,lazy,mid+1,high,qlow, qhigh,pos*2+2)); // returning the miniumum to the parent
}
void update(int seg[],int lazy[],int low,int high,int ulow,int uhigh,int delta ,int pos){
if(low>high)return;
if(lazy[pos]!=0){
seg[pos]+=lazy[pos]; //updating the segment tree with corresponding lazy
if(high!=low){
lazy[2*pos+1]+=lazy[pos]; //propagating to the left child
lazy[2*pos+2]+=lazy[pos]; // propagating to the right child
}
lazy[pos]=0;
}
if(ulow<=low&&uhigh>=high) // total overlapping
{
seg[pos]+=delta; // updating the parent child
if(high!=low){
lazy[2*pos+1]+=delta; // updating the left child
lazy[2*pos+2]+=delta; // updating the right child
}
return ;
}
if(ulow>high||uhigh<low)return ;
int mid=(high+low)/2;
update(seg,lazy,low,mid,ulow,uhigh,delta,2*pos+1);
update(seg,lazy,mid+1,high,ulow,uhigh,delta,2*pos+2);
seg[pos]=min(seg[pos*2+1],seg[2*pos+2]);
}
int main() {
int n,q;
scanf("%d%d",&n,&q);
int A[n];
int m=pow(2,1+ceil(log2(n)));
int seg[m],lazy[m];
for(int i=0;i<m;i++)lazy[i]=0;
for(int i=0;i<n;i++)scanf("%d",&A[i]);
build(A,seg,0,n-1,0);
int k,left,right,value;
while(q!=0){
scanf("%d%d%d%d",&k,&left,&right,&value);
if(k==1)
update(seg,lazy,0,n-1,left,right,value,0);
else if(k==2)
printf("%d \n",query(seg,lazy,0,n-1,left,right,0));
q--;
}
// for(int i=0;i<m;i++)printf("%d ",seg[i]);
return 0;
}