Дерево Фенвика — различия между версиями
| Строка 19: | Строка 19: | ||
| |statement= | |statement= | ||
| Нам надо научиться быстро изменять частичные суммы в зависимости от того, как изменяются элементы. Рассмотрим как изменять величину <tex>a_{k}</tex> на величину <tex>d</tex>. | Нам надо научиться быстро изменять частичные суммы в зависимости от того, как изменяются элементы. Рассмотрим как изменять величину <tex>a_{k}</tex> на величину <tex>d</tex>. | ||
| − | Необходимо изменить элементы дерева <tex>T_{i}</tex>, для которых верно неравенство <tex>F(i) \ | + | Необходимо изменить элементы дерева <tex>T_{i}</tex>, для которых верно неравенство <tex>F(i) \leqslant k \leqslant i</tex> . | 
| |proof= | |proof= | ||
| − | <tex> T_i =\sum\limits_{k = F(i)}^{i} a_k , i = 0 .. n - 1 \Rightarrow</tex> необходимо менять те <tex>i</tex>, для которых <tex>a_{k}</tex> попадает в <tex>T_i \Rightarrow</tex> необходимые <tex> i </tex> удовлетворяют условию <tex>F(i) \ | + | <tex> T_i =\sum\limits_{k = F(i)}^{i} a_k , i = 0 .. n - 1 \Rightarrow</tex> необходимо менять те <tex>i</tex>, для которых <tex>a_{k}</tex> попадает в <tex>T_i \Rightarrow</tex> необходимые <tex> i </tex> удовлетворяют условию <tex>F(i) \leqslant k \leqslant i</tex>. | 
| }} | }} | ||
| Строка 57: | Строка 57: | ||
| |statement= <tex> a_i </tex> входит в сумму для <tex> t_k </tex>, если <tex> \exists j: k = i \mid (2^j - 1) </tex>. | |statement= <tex> a_i </tex> входит в сумму для <tex> t_k </tex>, если <tex> \exists j: k = i \mid (2^j - 1) </tex>. | ||
| }} | }} | ||
| − | Для доказательства леммы рассмотрим битовую запись следующих чисел: <tex> k - 2^{h(k)} + 1 \ | + | Для доказательства леммы рассмотрим битовую запись следующих чисел: <tex> k - 2^{h(k)} + 1 \leqslant i \leqslant k </tex> | 
| {| style="background-color:#CCC;margin:0.5px" | {| style="background-color:#CCC;margin:0.5px" | ||
| Строка 90: | Строка 90: | ||
| * [http://en.wikipedia.org/wiki/ Wikipedia — Fenwick_tree Fenwick tree] | * [http://en.wikipedia.org/wiki/ Wikipedia — Fenwick_tree Fenwick tree] | ||
| * [http://e-maxx.ru/algo/fenwick_tree e-maxx.ru — Дерево Фенвика] | * [http://e-maxx.ru/algo/fenwick_tree e-maxx.ru — Дерево Фенвика] | ||
| + | |||
| + | |||
| + | [[Категория: Дискретная математика и алгоритмы]] | ||
| + | |||
| + | [[Категория: Дерево Фенвика]] | ||
Версия 15:55, 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;
}

