Дерево Фенвика — различия между версиями
(→Запрос изменения элемента) |
|||
Строка 16: | Строка 16: | ||
== Запрос изменения элемента == | == Запрос изменения элемента == | ||
− | Нам надо научиться быстро изменять частичные суммы в зависимости от того, как изменяются элементы. Рассмотрим как изменять величину <tex>a_{k}</tex> на величину <tex>d</tex>. Тогда нам надо изменить элементы дерева <tex>T_{i}</tex>, для которых верно неравенство <tex>F(i) < k < i</tex>. Все <tex>i</tex> мы можем получить следующим образом : <tex>i_{next} = i_{prev} | (i_{prev} + 1) </tex>, Где под | понимают побитовое ИЛИ. Для того, чтобы понять, что эта последовательность верна, достаточно посмотреть на | + | Нам надо научиться быстро изменять частичные суммы в зависимости от того, как изменяются элементы. Рассмотрим как изменять величину <tex>a_{k}</tex> на величину <tex>d</tex>. Тогда нам надо изменить элементы дерева <tex>T_{i}</tex>, для которых верно неравенство <tex>F(i) < k < i</tex>. Все <tex>i</tex> мы можем получить следующим образом : <tex>i_{next} = i_{prev} | (i_{prev} + 1) </tex>, Где под | понимают побитовое ИЛИ. Для того, чтобы понять, что эта последовательность верна, достаточно посмотреть на таблицу. |
+ | |||
+ | {| border="1" | ||
+ | |<tex>i_{prev}</tex> | ||
+ | |<tex>\cdots 011 \cdots 1)</tex> | ||
+ | |- | ||
+ | |<tex>i_{prev} + 1</tex> | ||
+ | |<tex>\cdots 100 \cdots 0)</tex> | ||
+ | |- | ||
+ | |<tex>i_{next}</tex> | ||
+ | |<tex>\cdots 111 \cdots 1)</tex> | ||
+ | |} | ||
== Запрос получения суммы на префиксе == | == Запрос получения суммы на префиксе == |
Версия 23:32, 8 мая 2011
Определение: |
Дерево Фе́нвика (Binary indexed tree) - структура данных, требующая
| памяти и позволяющая эффективно (за )
Впервые описано Питером Фенвиком в 1994 году.
Пусть дан массив
Деревом Фенвика будем называть массив из элементов: , где - некоторая функция.
От выбора функции зависит время работы операций над деревом. Рассмотрим функцию, позволяющую делать обе операции за время .
где - количество единиц в конце бинарной записи числа . Эта функция задается простой формулой: .
Содержание
Запрос изменения элемента
Нам надо научиться быстро изменять частичные суммы в зависимости от того, как изменяются элементы. Рассмотрим как изменять величину
на величину . Тогда нам надо изменить элементы дерева , для которых верно неравенство . Все мы можем получить следующим образом : , Где под | понимают побитовое ИЛИ. Для того, чтобы понять, что эта последовательность верна, достаточно посмотреть на таблицу.Запрос получения суммы на префиксе
В качестве бинарной операции
Обозначим . Тогда .
Лемма: |
входит в сумму для , если . |
Для доказательства леммы рассмотрим битовую запись следующих чисел:
Реализация
Приведем код функции
на 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: Дерево Фенвика