Реализация запроса в дереве отрезков снизу — различия между версиями
Shersh (обсуждение | вклад) м (→Псевдокод) |
(→Псевдокод: ошибка обновления левой границы ((13 - 1) / 2 почему то 5... непорядок)) |
||
Строка 33: | Строка 33: | ||
'''if''' right '''mod''' 2 == 1 | '''if''' right '''mod''' 2 == 1 | ||
rightRes = data[right] <tex> \circ </tex> rightRes | rightRes = data[right] <tex> \circ </tex> rightRes | ||
− | right = | + | right = right '''div''' 2 - 1 |
'''if''' left == right | '''if''' left == right | ||
leftRes = leftRes <tex> \circ </tex> data[left] | leftRes = leftRes <tex> \circ </tex> data[left] |
Версия 23:36, 22 марта 2018
Содержание
Алгоритм
Реализация запроса снизу вверх в дереве отрезков является, в отличие от реализации запроса сверху вниз, итеративным методом. Будем рассматривать абстрактную операцию, обладающую свойством ассоциативности, и обозначать ее .
Построим дерево отрезков и установим границы отрезка на соответствующие им листья. Будем действовать в 3 этапа:
- Если элемент, попавший на левую границу, является правым сыном, то запишем в результат значение, полученное после выполнения нашей операции над предыдущим результатом и значением этого элемента, а левую границу перемещаем на один элемент вправо. Аналогично с правой границей (является ли она левым сыном). Таким образом мы учтем вклад нужной нам вершины и избавимся от вклада ненужного нам поддерева.
- Устанавливаем границы отрезка на родительские элементы текущих границ. Это позволит узнать, входит ли полученный отрезок в искомый или нет. Повторяем этап 1, пока границы не пересекутся.
- Если после завершения цикла границы совпадут, значит полученный отрезок входит в искомый, и надо пересчитать результат.
Реализация запроса снизу вверх | Ещё один пример |
Псевдокод
Пусть результат считаем на отрезке
. При этом значения и , передающиеся в функцию, должны указывать на листья дерева (необходимо увеличить значение на индекс массива, с которого начинаются листья). Переменные и будут собирать значения на отрезках, отделившихся соответственно слева или справа от рассматриваемого.int query(left: int, right: int): leftRes = neutral rightRes = neutral while left < right if left mod 2 == 0 leftRes = leftResdata[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