KEMBAR78
Segment Tree | PDF | Matrix (Mathematics) | Pointer (Computer Programming)
0% found this document useful (0 votes)
173 views36 pages

Segment Tree

A segment tree is a data structure that allows efficient range queries on an array. It supports queries like finding the sum of elements in a range in O(logn) time. The segment tree is built as a binary tree where each node represents a subarray (or segment). Precomputed values like sums of elements are stored in each node to answer queries quickly by combining stored values. Range update operations like adding a value to elements in a segment can also be supported in O(logn) time using lazy propagation, where updates are postponed and propagated on query/update.

Uploaded by

qweqwe
Copyright
© © All Rights Reserved
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as PPTX, PDF, TXT or read online on Scribd
0% found this document useful (0 votes)
173 views36 pages

Segment Tree

A segment tree is a data structure that allows efficient range queries on an array. It supports queries like finding the sum of elements in a range in O(logn) time. The segment tree is built as a binary tree where each node represents a subarray (or segment). Precomputed values like sums of elements are stored in each node to answer queries quickly by combining stored values. Range update operations like adding a value to elements in a segment can also be supported in O(logn) time using lazy propagation, where updates are postponed and propagated on query/update.

Uploaded by

qweqwe
Copyright
© © All Rights Reserved
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as PPTX, PDF, TXT or read online on Scribd
You are on page 1/ 36

Segment Tree

Introduction
• a data structure that allows answering range queries over an array.
• the sum of consecutive array elements in .
• the minimum element in a such a range in  time.
• and allows modifying the array by replacing one element, or even
change the elements of a whole subsegment.
• assign all elements in to any value.
• add a value to all element in the subsegment.

2
Introduction
• a very flexible data structure which a huge number of problems can be
solved with it.
• answer sum or minimum queries over some subrectangle of a given matrix
with a two-dimensional Segment Tree.
• The standard Segment Tree requires vertices for working on an array
of size .

3
a Segment Tree to answer sum queries
• find the in .
• chang values of the elements in the array ().
• process both queries in .

4
Structure of Segment Tree
• a binary tree: the root of tree is
the segment , and each vertex
(except leaf vertices) has exactly
two child vertices which is
and ,and so on.  
• For each such segment we store
the sum of the numbers on it.

a Segment Tree over the array 

5
Structure of Segment Tree
• Segment Tree only requires a
linear number of vertices.
• the number of vertices in the
worst case can be estimated by
the sum .
• not all levels of the Segment
Tree will be completely filled.
• The height is .
a Segment Tree over the array 

6
store the Segment Tree
• define a Vertex struct and create objects, that store the boundaries of
the segment, its sum and additionally also pointers to its child vertices.
However this requires storing a lot of redundant information.
• only store the sums in an array, that means the left child of a vertex at
index is stored at index , and the right one at index .
• store the Segment Tree simply as an array with a size of four times the
input size :
int n, t[4*MAXN];

7
Construction
• start at the bottom level, the leaf vertices and simply copy the values
of the elements .
• compute the sums of the previous level. 
• repeat the procedure until we reach the root vertex.
• The time complexity of the construction is .

8
Construction-code
void build(int a[], int v, int tl, int tr) {
if (tl == tr) {
t[v] = a[tl];
} else {
int tm = (tl + tr) / 2;
build(a, v*2, tl, tm);
build(a, v*2+1, tm+1, tr);
t[v] = t[v*2] + t[v*2+1];
}
}

9
Sum queries
• compute in .
• traverse the Segment Tree and use the precomputed sums of the
segments.
• at the vertex that covers the segment ,There are three possible cases:

1. or ,

10
Sum queries
3. the query segment intersects with both children.
make two recursive calls, one for each child.

• processing a sum query is a function that


recursively calls itself once with either the left
or the right child.
• the calculation of the query is a traversal of
the tree, which spreads through all necessary
branches of the tree, and uses the
precomputed sum values of the segments in
the tree. compute the sum

11
Sum queries
• the complexity of this algorithm is .
• the query works by dividing the input
segment into several sub-segments for
which all the sums are already
precomputed and stored in the tree.

compute the sum

12
Sum queries - code
int sum(int v, int tl, int tr, int l, int r) {
if (l > r)
return 0;
if (l == tl && r == tr) {
return t[v];
}
int tm = (tl + tr) / 2;
return sum(v*2, tl, tm, l, min(r, tm))
+ sum(v*2+1, tm+1, tr, max(l, tm+1), r);
}

13
Update queries
• do the assignment 
• an element  only contributes to one
segment from each level. Thus only
vertices need to be updated.

update

14
Update queries - code
void update(int v, int tl, int tr, int pos, int new_val) {
if (tl == tr) {
t[v] = new_val;
} else {
int tm = (tl + tr) / 2;
if (pos <= tm)
update(v*2, tl, tm, pos, new_val);
else
update(v*2+1, tm+1, tr, pos, new_val);
t[v] = t[v*2] + t[v*2+1];
}
}

15
Memory efficient implementation
• The memory consumption is limited by 4n, even though a Segment
Tree of an array of n elements requires only 2n minus 1 vertices.
• renumber the vertices of the tree in the order of an Euler tour traversal
(pre-order traversal), and we write all these vertices next to each other.
• a vertex is responsible for the segment , the left child have the index ,
in total there will be vertices in the left child's subtree. and the index
of the right child of will be .

16
Range updates (Lazy Propagation)
• the Segment Tree allows applying modification queries to an entire
segment of contiguous elements and perform the query in the same
time .
• Addition on segments
1. add a number  to all numbers in the segment .
2. answer the value of .
• store at each vertex in the Segment Tree how many we should add to
all numbers in the corresponding segment. 
• don't have to change all  values, but only many.

17
Addition on segments- code
void build(int a[], int v, int tl, int tr) {
if (tl == tr) t[v] = a[tl];
else {
int tm = (tl + tr) / 2;
build(a, v*2, tl, tm);
build(a, v*2+1, tm+1, tr);
t[v] = 0;
}
}
void update(int v, int tl, int tr, int l, int r, int add) {
if (l > r) return;
if (l == tl && r == tr) t[v] += add;
else {
int tm = (tl + tr) / 2;
update(v*2, tl, tm, l, min(r, tm), add);
update(v*2+1, tm+1, tr, max(l, tm+1), r, add);
}
}
18
Addition on segments - code
int get(int v, int tl, int tr, int pos) {
if (tl == tr)
return t[v];
int tm = (tl + tr) / 2;
if (pos <= tm)
return t[v] + get(v*2, tl, tm, pos);
else
return t[v] + get(v*2+1, tm+1, tr, pos);
}

19
Range updates (Lazy Propagation)
• Assignment on segments
1. assign each element of a certain segment to some value .
2. answer the value of .
• "lazy" update: instead of changing all segments in the tree that cover
the query segment, we only change some, and leave others unchanged.
• we are lazy and delay writing the new value to all those vertices. We
can do this tedious task later, if this is necessary.

20
Range updates (Lazy Propagation)
• Query : assign a number to the whole array a[0…n−1]a[0…n−1].
• Change : the number is placed in the root of the tree and this vertex
gets marked. The remaining segments remain unchanged.
• Query : the first half of the array  should be assigned with some other
number.
• Change:
1. push the information of the root to its children.
2. assign the left and the right child vertices with this number and remove the
mark of the root.

21
Range updates (Lazy Propagation)
• for any queries (a modification or reading query) during the descent
along the tree we should always push information from the current
vertex into both of its children. 
• implementation :   function, which will receive the current vertex, and
it will push the information for its vertex to both its children. 

22
Assignment on segments - code
void push(int v) {
if (marked[v]) {
t[v*2] = t[v*2+1] = t[v];
marked[v*2] = marked[v*2+1] = true;
marked[v] = false;
}
}
void update(int v, int tl, int tr, int l, int r, int new_val) {
if (l > r) return;
if (l == tl && tr == r) {
t[v] = new_val;
marked[v] = true;
} else {
push(v);
int tm = (tl + tr) / 2;
update(v*2, tl, tm, l, min(r, tm), new_val);
update(v*2+1, tm+1, tr, max(l, tm+1), r, new_val);
}
}
23
Assignment on segments - code
int get(int v, int tl, int tr, int pos) {
if (tl == tr) {
return t[v];
}
push(v);
int tm = (tl + tr) / 2;
if (pos <= tm)
return get(v*2, tl, tm, pos);
else
return get(v*2+1, tm+1, tr, pos);
}

24
Range updates (Lazy Propagation)
• Query : Adding on segments, querying for maximum
• store the maximum of the corresponding subsegment for each vertex.
• how to recompute these values during a modification request ?
• Before traversing to a child vertex, we call  and propagate the value to both
children in both the  and the .

25
Adding on segments , querying for maximum - code

void push(int v) {
t[v*2] += lazy[v];
lazy[v*2] += lazy[v];
t[v*2+1] += lazy[v];
lazy[v*2+1] += lazy[v];
lazy[v] = 0;
}
void update(int v, int tl, int tr, int l, int r, int addend) {
if (l > r) return;
if (l == tl && tr == r) {
t[v] += addend; lazy[v] += addend;
} else {
push(v);
int tm = (tl + tr) / 2;
update(v*2, tl, tm, l, min(r, tm), addend);
update(v*2+1, tm+1, tr, max(l, tm+1), r, addend);
t[v] = max(t[v*2], t[v*2+1]);
}
}
26
Adding on segments , querying for maximum - code

int query(int v, int tl, int tr, int l, int r) {


if (l > r)
return -INF;
if (l <= tl && tr <= r)
return t[v];
push(v);
int tm = (tl + tr) / 2;
return max(query(v*2, tl, tm, l, min(r, tm)),
query(v*2+1, tm+1, tr, max(l, tm+1), r));
}

27
Persistent Segment Tree
• A persistent data structure is a data structure that remembers it
previous state for each modification.
• turn regular Segment Tree into a persistent data structure
• avoid copying the complete tree before each modification.
• keep the time behavior for answering range queries.

28
Persistent Segment Tree
• store the Segment Tree using pointers.
• create new vertices instead of changing the available vertices.
• new vertices will be created, including a new root vertex of the
Segment Tree. [ 1,10 ] [ 1,10 ]
[ 1,5 ] [ 6,10 ] [ 6,10 ]

[ 1,3 ] [ 4,5 ] [ 6,8 ] [ 9,10 ] [ 6,8 ]

[ 1,2 ] [ 3,3 ] [ 4,4 ] [ 5,5 ] [ 6,7 ] [ 8,8 ] [ 9,9 ] [ 10,10 ] [ 6,7 ]

[ 1,1 ] [ 2,2 ] [ 6,6 ] [ 7,7 ] [ 7,7 ]


29
Persistent Segment Tree - code
struct Vertex {
Vertex *l, *r;
int sum;
Vertex(int val) : l(nullptr), r(nullptr), sum(val) {}
Vertex(Vertex *l, Vertex *r) : l(l), r(r), sum(0) {
if (l) sum += l->sum;
if (r) sum += r->sum;
}
};

Vertex* build(int a[], int tl, int tr) {


if (tl == tr)
return new Vertex(a[tl]);
int tm = (tl + tr) / 2;
return new Vertex(build(a, tl, tm), build(a, tm+1, tr));
}

30
Persistent Segment Tree - code
int get_sum(Vertex* v, int tl, int tr, int l, int r) {
if (l > r)
return 0;
if (l == tl && tr == r)
return v->sum;
int tm = (tl + tr) / 2;
return get_sum(v->l, tl, tm, l, min(r, tm))
+ get_sum(v->r, tm+1, tr, max(l, tm+1), r);
}

Vertex* update(Vertex* v, int tl, int tr, int pos, int new_val) {
if (tl == tr)
return new Vertex(new_val);
int tm = (tl + tr) / 2;
if (pos <= tm)
return new Vertex(update(v->l, tl, tm, pos, new_val), v->r);
else
return new Vertex(v->l, update(v->r, tm+1, tr, pos, new_val));
}
31
Finding the k-th smallest number in a range
• write a new data structure that would be able to return quickly k-th
order statistics in the array segment. That is, given an array of different
integer numbers, your program must answer a series of questions in
the form: "What would be the k-th number in segment, if this segment
was sorted?" 
For example, consider the array . Let the question be . The segment
is . If we sort this segment, we get , the third number is , and therefore
the answer to the question is .
•   : the size of the array, : the number of questions to answer (). 

32
Finding the k-th smallest number in a range
• a solution for a simpler problem:
• only consider arrays in which the elements are bound by .
• only want to find the k-th smallest element in some prefix of the array .
• use a Segment Tree that counts all appearing numbers.
• we will create persistent Segment Trees one by one :
• start with an empty Segment Tree (all counts will be 0),
• add the elements one after each other.
• For each modification we will receive a new root vertex, ,the root of the
Segment Tree after inserting .
• The Segment Tree rooted at will contain the histogram of the prefix .
33
Finding the k-th smallest number in a range
• the restriction on the array: transform it by index compression.
• the queries restriction: use any segments .
such a Segment Tree is just the difference between the Segment Tree
rooted at and the Segment Tree rooted at .
• In the implementation
pass two vertex pointer and computing the count/sum of the current
segment as difference of the two counts/sums of the vertices.

34
Implicit Segment Tree
• what to do if the the size of a segment tree does not allow you to
completely build up to it in advance?
• soluton:
1. create only the root, and the other vertexes only when we need them.
2. use the implementation on pointers(before going to the vertex children,
check whether they are created, and if not, create them).
3. Each query has still only the complexity .
• two queries
1. adding a value to a position (initially all values are 0)
2. computing the sum of all values in a range.
• Vertex(0, n) will be the root vertex of the implicit tree.
35
The End

36

You might also like