Реализация запроса в дереве отрезков снизу — различия между версиями
Lirik (обсуждение | вклад) |
(Правка пунктуации) |
||
| (не показаны 32 промежуточные версии 8 участников) | |||
| Строка 1: | Строка 1: | ||
| − | + | ==Алгоритм== | |
| − | + | Реализация запроса снизу вверх в дереве отрезков является, в отличие от [[Реализация запроса в дереве отрезков сверху| реализации запроса сверху вниз]], итеративным методом. Будем рассматривать абстрактную операцию, обладающую свойством ассоциативности, и обозначать ее <tex>a \circ b</tex>. | |
| − | + | [[Дерево отрезков. Построение|Построим дерево отрезков]], и установим границы отрезка на соответствующие им листья. Будем действовать в 3 этапа: | |
| + | # Если элемент, попавший на левую границу, является правым сыном, то запишем в результат значение, полученное после выполнения нашей операции над предыдущим результатом и значением этого элемента, а левую границу перемещаем на один элемент вправо. Аналогично с правой границей (является ли она левым сыном). Таким образом мы учтем вклад нужной нам вершины и избавимся от вклада ненужного нам поддерева. | ||
| + | # Устанавливаем границы отрезка на родительские элементы текущих границ. Это позволит узнать, входит ли полученный отрезок в искомый или нет. Повторяем этап 1, пока границы не пересекутся. | ||
| + | # Если после завершения цикла границы совпадут, значит полученный отрезок входит в искомый, и надо пересчитать результат. | ||
| − | + | {| cellpadding="10" | |
| + | | '''Реализация запроса снизу вверх''' || '''Ещё один пример''' | ||
| + | |- | ||
| + | | [[Файл:Запрос снизу №1х1.jpg|550px|]] || [[Файл:Запрос снизу №2х1.jpg|550px]] | ||
| + | |- | ||
| + | | [[Файл:Запрос снизу №1х2.jpg|550px|]] || [[Файл:Запрос снизу №2х2.jpg|550px]] | ||
| + | |- | ||
| + | | [[Файл:Запрос снизу №1х3.jpg|550px|]] || [[Файл:Запрос снизу №2х3.jpg|550px]] | ||
| + | |- | ||
| + | | [[Файл:Запрос снизу №1х4.jpg|550px|]] || [[Файл:Запрос снизу №2х4.jpg|550px]] | ||
| + | |} | ||
==Псевдокод== | ==Псевдокод== | ||
| − | Пусть | + | Пусть результат считаем на отрезке <tex> [left, right] </tex>. При этом значения <tex>left</tex> и <tex>right</tex>, передающиеся в функцию, должны указывать на листья дерева (необходимо увеличить значение на индекс массива, с которого начинаются листья). Переменные <tex>leftRes</tex> и <tex>rightRes</tex> будут собирать значения на отрезках, отделившихся соответственно слева или справа от рассматриваемого. |
| + | |||
| + | '''int''' query(left: '''int''', right: '''int'''): | ||
| + | leftRes = ''neutral'' | ||
| + | rightRes = ''neutral'' | ||
| + | '''while''' left < right | ||
| + | '''if''' left '''mod''' 2 == 0 | ||
| + | leftRes = leftRes <tex> \circ </tex> data[left] | ||
| + | left = left '''div''' 2 | ||
| + | '''if''' right '''mod''' 2 == 1 | ||
| + | rightRes = data[right] <tex> \circ </tex> rightRes | ||
| + | right = right '''div''' 2 - 1 | ||
| + | '''if''' left == right | ||
| + | leftRes = leftRes <tex> \circ </tex> data[left] | ||
| + | '''return''' leftRes <tex> \circ </tex> rightRes | ||
| + | |||
| + | ==См. также== | ||
| + | * [[Реализация запроса в дереве отрезков сверху]] | ||
| + | |||
| + | ==Источники информации== | ||
| − | + | * [http://e-maxx.ru/algo/segment_tree MAXimal :: algo :: Дерево отрезков] | |
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | + | * [http://ru.wikipedia.org/wiki/%D0%94%D0%B5%D1%80%D0%B5%D0%B2%D0%BE_%D0%BE%D1%82%D1%80%D0%B5%D0%B7%D0%BA%D0%BE%D0%B2 Википедия {{---}} Дерево отрезков] | |
| − | [http:// | + | * [http://rain.ifmo.ru/cat/view.php/vis/trees/segment-2006 Визуализатор] |
| − | [http://ru. | + | * [http://rain.ifmo.ru/cat/view.php/vis/trees/segment-2006/algorithm Алгоритм] |
| − | [ | + | [[Категория: Дискретная математика и алгоритмы]] |
| − | [ | + | [[Категория: Дерево отрезков]] |
Версия 22:16, 5 сентября 2019
Содержание
Алгоритм
Реализация запроса снизу вверх в дереве отрезков является, в отличие от реализации запроса сверху вниз, итеративным методом. Будем рассматривать абстрактную операцию, обладающую свойством ассоциативности, и обозначать ее .
Построим дерево отрезков, и установим границы отрезка на соответствующие им листья. Будем действовать в 3 этапа:
- Если элемент, попавший на левую границу, является правым сыном, то запишем в результат значение, полученное после выполнения нашей операции над предыдущим результатом и значением этого элемента, а левую границу перемещаем на один элемент вправо. Аналогично с правой границей (является ли она левым сыном). Таким образом мы учтем вклад нужной нам вершины и избавимся от вклада ненужного нам поддерева.
- Устанавливаем границы отрезка на родительские элементы текущих границ. Это позволит узнать, входит ли полученный отрезок в искомый или нет. Повторяем этап 1, пока границы не пересекутся.
- Если после завершения цикла границы совпадут, значит полученный отрезок входит в искомый, и надо пересчитать результат.
| Реализация запроса снизу вверх | Ещё один пример |
|
|
|
|
|
|
|
|
Псевдокод
Пусть результат считаем на отрезке . При этом значения и , передающиеся в функцию, должны указывать на листья дерева (необходимо увеличить значение на индекс массива, с которого начинаются листья). Переменные и будут собирать значения на отрезках, отделившихся соответственно слева или справа от рассматриваемого.
int query(left: int, right: int):
leftRes = neutral
rightRes = neutral
while left < right
if left mod 2 == 0
leftRes = leftRes data[left]
left = left div 2
if right mod 2 == 1
rightRes = data[right] rightRes
right = right div 2 - 1
if left == right
leftRes = leftRes data[left]
return leftRes rightRes







