Дерево Фенвика — различия между версиями
| Строка 20: | Строка 20: | ||
{| border="1" | {| border="1" | ||
|<tex>i_{prev}</tex> | |<tex>i_{prev}</tex> | ||
| − | |<tex>\cdots 011 \cdots 1 | + | |<tex>\cdots 011 \cdots 1</tex> |
|- | |- | ||
|<tex>i_{prev} + 1</tex> | |<tex>i_{prev} + 1</tex> | ||
| − | |<tex>\cdots 100 \cdots 0 | + | |<tex>\cdots 100 \cdots 0</tex> |
|- | |- | ||
|<tex>i_{next}</tex> | |<tex>i_{next}</tex> | ||
| − | |<tex>\cdots 111 \cdots 1 | + | |<tex>\cdots 111 \cdots 1</tex> |
|} | |} | ||
| + | |||
| + | |||
| + | Несложно заметить, что данная последовательность строго возрастает и в худшем случае будет применена логарифм раз, так как добавляет каждый раз по одной единице в двоичном разложении числа <tex>i</tex>. | ||
| + | Напишем функцию, которая будет изменять <tex>a_i</tex> элемент на <tex>d</tex>, и при этом меняет соответствующие частичные суммы. | ||
| + | |||
| + | int modify(int i, int d) | ||
| + | { | ||
| + | while (i < N) | ||
| + | { | ||
| + | t[i] += d; | ||
| + | i = i | (i + 1); | ||
| + | } | ||
| + | } | ||
| + | |||
== Запрос получения суммы на префиксе == | == Запрос получения суммы на префиксе == | ||
Версия 23:38, 8 мая 2011
| Определение: |
Дерево Фе́нвика (Binary indexed tree) - структура данных, требующая памяти и позволяющая эффективно (за )
|
Впервые описано Питером Фенвиком в 1994 году.
Пусть дан массив из элементов: .
Деревом Фенвика будем называть массив из элементов: , где - некоторая функция.
От выбора функции зависит время работы операций над деревом. Рассмотрим функцию, позволяющую делать обе операции за время .
где - количество единиц в конце бинарной записи числа . Эта функция задается простой формулой: .
Содержание
Запрос изменения элемента
Нам надо научиться быстро изменять частичные суммы в зависимости от того, как изменяются элементы. Рассмотрим как изменять величину на величину . Тогда нам надо изменить элементы дерева , для которых верно неравенство . Все мы можем получить следующим образом : , Где под | понимают побитовое ИЛИ. Для того, чтобы понять, что эта последовательность верна, достаточно посмотреть на таблицу.
Несложно заметить, что данная последовательность строго возрастает и в худшем случае будет применена логарифм раз, так как добавляет каждый раз по одной единице в двоичном разложении числа .
Напишем функцию, которая будет изменять элемент на , и при этом меняет соответствующие частичные суммы.
int modify(int i, int 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;
}
Полезные ссылки:
Peter M. Fenwick: A new data structure for cumulative frequency
Wikipedia: Fenwick tree
e-maxx.ru: Дерево Фенвика
