Дерево Фенвика — различия между версиями
| Строка 52: | Строка 52: | ||
== Запрос получения суммы на префиксе == | == Запрос получения суммы на префиксе == | ||
В качестве бинарной операции <tex> G </tex> рассмотрим операцию сложения. <br/> | В качестве бинарной операции <tex> G </tex> рассмотрим операцию сложения. <br/> | ||
| − | Обозначим <tex> G_i = sum(i) = \sum\limits_{k = 0}^{i} a_k </tex>. Тогда <tex> sum(i, j) = \sum\limits_{k = i}^{j} a_k = G_j - G_{i - 1} </tex>. | + | Обозначим <tex> G_i = \mathrm sum(i) = \sum\limits_{k = 0}^{i} a_k </tex>. Тогда <tex> \mathrm sum(i, j) = \sum\limits_{k = i}^{j} a_k = G_j - G_{i - 1} </tex>. |
{{Лемма | {{Лемма | ||
| Строка 71: | Строка 71: | ||
=== Реализация === | === Реализация === | ||
| − | Приведем код функции <tex> sum(i) </tex> на C++: | + | Приведем код функции <tex> \mathrm sum(i) </tex> на C++: |
<code> | <code> | ||
int sum(int i) | int sum(int i) | ||
Версия 16:02, 26 марта 2015
| Определение: |
Дерево Фе́нвика (Binary indexed tree) — структура данных, требующая памяти и позволяющая эффективно (за )
|
Впервые описано Питером Фенвиком в 1994 году.
Пусть дан массив из элементов: .
Деревом Фенвика будем называть массив из элементов: , где - некоторая функция.
От выбора функции зависит время работы операций над деревом. Рассмотрим функцию, позволяющую делать обе операции за время .
где - количество единиц в конце бинарной записи числа . Эта функция задается простой формулой: .
Содержание
Запрос изменения элемента
| Лемма: |
Нам надо научиться быстро изменять частичные суммы в зависимости от того, как изменяются элементы. Рассмотрим как изменять величину на величину .
Необходимо изменить элементы дерева , для которых верно неравенство . |
| Доказательство: |
| необходимо менять те , для которых попадает в необходимые удовлетворяют условию . |
| Лемма: |
Можно перебрать все , попадающие под неравенство по формуле . |
| Доказательство: |
| Первый элемент последовательности само . Для него выполняется равенство, так как . По формуле мы заменим первый ноль на единицу. Неравенство при этом сохранится, так как осталось прежним, а увеличилось. Можем заметить, что если количество единиц в конце не будет совпадать с , то формула нарушит неравенство, потому что либо само будет меньше, чем k, либо станет больше, чем . Таким образом, перебраны будут только нужные элементы |
Все мы можем получить следующим образом : , Где под | понимают побитовое ИЛИ. Следующим элементом в последовательности будет элемент, у которого первый с конца ноль превратится в единицу. Можно заметить, что если к исходному элементу прибавить единицу, то необходимый ноль обратится в единицу, но при этом все следующие единицы обнулятся. Чтобы обратно их превратить в единицы, применим операцию побитового ИЛИ. Таким образом все нули в конце превратятся в единицы и мы получим нужный элемент. Для того, чтобы понять, что эта последовательность верна, достаточно посмотреть на таблицу.
Несложно заметить, что данная последовательность строго возрастает и в худшем случае будет применена логарифм раз, так как добавляет каждый раз по одной единице в двоичном разложении числа .
Напишем функцию, которая будет изменять элемент на , и при этом меняет соответствующие частичные суммы.
modify(i, d):
while i < N
t[i] += d
i = i | (i + 1)
Запрос получения суммы на префиксе
В качестве бинарной операции рассмотрим операцию сложения.
Обозначим . Тогда .
| Лемма: |
входит в сумму для , если . |
Для доказательства леммы рассмотрим битовую запись следующих чисел:
Реализация
Приведем код функции на C++:
int sum(int i)
{
int result = 0;
while (i >= 0)
{
result += t[i];
i = f(i) - 1;
}
return result;
}
