Дерево Фенвика — различия между версиями
Shersh (обсуждение | вклад) м |
|||
| Строка 29: | Строка 29: | ||
{| style="background-color:#CCC;margin:0.5px" | {| style="background-color:#CCC;margin:0.5px" | ||
| − | |style="background-color:#EEE;padding: | + | |style="background-color:#EEE;padding:3px 10px"| <tex>i</tex>, десятичная запись |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>0</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>1</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>2</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>3</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>4</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>5</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>6</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>7</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>8</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>9</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>10</tex> |
|- | |- | ||
| − | |style="background-color:#EEE;padding: | + | |style="background-color:#EEE;padding:3px 10px"| <tex>i</tex>, двоичная запись |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0000</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0001</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0010</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0011</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0100</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0101</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0110</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0111</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>1000</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>1001</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>1010</tex> |
|- | |- | ||
| − | |style="background-color:#EEE;padding: | + | |style="background-color:#EEE;padding:3px 10px"| <tex>F(i)</tex>, двоичная запись |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0000</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0000</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0010</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0000</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0100</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0100</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0110</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>0000</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>1000</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>1000</tex> |
| − | |style="background-color:#FFF | + | |style="background-color:#FFF"| <tex>1010</tex> |
|- | |- | ||
| − | |style="background-color:#EEE;padding: | + | |style="background-color:#EEE;padding:3px 10px"| <tex>F(i)</tex>, десятичная запись |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>0</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>0</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>2</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>0</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>4</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>4</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>6</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>0</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>8</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>8</tex> |
| − | |style="background-color:#FFF;padding: | + | |style="background-color:#FFF;padding:3px 10px"| <tex>10</tex> |
|} | |} | ||
| Строка 95: | Строка 95: | ||
|style="background-color:#FFF;padding:2px 30px"| <tex>\ldots 111 \ldots 1</tex> | |style="background-color:#FFF;padding:2px 30px"| <tex>\ldots 111 \ldots 1</tex> | ||
|} | |} | ||
| − | |||
Несложно заметить, что данная последовательность строго возрастает и в худшем случае будет применена логарифм раз, так как добавляет каждый раз по одной единице в двоичном разложении числа <tex>i</tex>. | Несложно заметить, что данная последовательность строго возрастает и в худшем случае будет применена логарифм раз, так как добавляет каждый раз по одной единице в двоичном разложении числа <tex>i</tex>. | ||
| Строка 109: | Строка 108: | ||
В качестве бинарной операции <tex> \circ </tex> рассмотрим операцию сложения. <br/> | В качестве бинарной операции <tex> \circ </tex> рассмотрим операцию сложения. <br/> | ||
| − | Обозначим <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>. |
{{Лемма | {{Лемма | ||
| Строка 136: | Строка 135: | ||
'''return''' result | '''return''' result | ||
| − | == | + | ==Сравнение дерева Фенвика и дерева отрезков== |
| − | + | * И дерево Фенвика, и дерево отрезков занимает <tex>O(n)</tex> памяти, но в случае, когда <tex> n = a^k + 1</tex>, дерево Фенвика занимает в 2 раза меньше памяти, чем дерево отрезков. | |
| − | + | * Дерево Фенвика проще в реализации. | |
| − | + | * Функция, для которой строится дерево Фенвика, должна быть обратимой, а это значит, что минимум и максимум это дерево считать не может (за исключением случаев, когда некоторыми данными мы можем пожертвовать), в отличие от дерева отрезков. | |
== См. также == | == См. также == | ||
| Строка 149: | Строка 148: | ||
* [http://en.wikipedia.org/wiki/Fenwick_tree Wikipedia — Fenwick_tree Fenwick tree] | * [http://en.wikipedia.org/wiki/Fenwick_tree Wikipedia — Fenwick_tree Fenwick tree] | ||
* [http://e-maxx.ru/algo/fenwick_tree Maximal:: algo:: Дерево Фенвика] | * [http://e-maxx.ru/algo/fenwick_tree Maximal:: algo:: Дерево Фенвика] | ||
| + | * [http://habrahabr.ru/post/112828 Хабрахабр - Дерево Фенвика] | ||
| Строка 154: | Строка 154: | ||
[[Категория: Дерево Фенвика]] | [[Категория: Дерево Фенвика]] | ||
| + | |||
| + | [[Категория: Структуры данных]] | ||
Версия 16:02, 29 мая 2015
Содержание
Описание структуры
Дерево Фе́нвика (англ. Binary indexed tree) — структура данных, требующая памяти и позволяющая эффективно (за ) выполнять следующие операции:
- изменять значение любого элемента в массиве,
- выполнять некоторую ассоциативную, коммутативную, обратимую операцию на отрезке .
Впервые описано Питером Фенвиком в 1994 году.
Пусть дан массив . Деревом Фенвика будем называть массив из элементов: , где и — некоторая функция, от выбора которой зависит время работы операций над деревом. Рассмотрим функцию, позволяющую делать операции вставки и изменения элемента за время . Она задается простой формулой: , где — это операция побитового логического . При числа и его значения, увеличенного на единицу, мы получаем это число без последних подряд идущих единиц.
Эту функцию можно вычислять по другой формуле: где — количество подряд идущих единиц в конце бинарной записи числа . Оба варианта равносильны, так как функция, заданная какой-либо из этих формул, заменяет все подряд идущие единицы в конце числа на нули.
Запрос изменения элемента
Нам надо научиться быстро изменять частичные суммы в зависимости от того, как изменяются элементы. Рассмотрим как изменяется массив при изменении элемента .
| Лемма: |
Для изменения величины необходимо изменить элементы дерева , для индексов которых верно неравенство . |
| Доказательство: |
| необходимо менять те , для которых попадает в необходимые удовлетворяют условию . |
| Лемма: |
Все такие можно найти по формуле , где — это операция побитового логического . |
| Доказательство: |
| Из доказанной выше леммы следует, что первый элемент последовательности само . Для него выполняется равенство, так как . По формуле мы заменим первый ноль на единицу. Неравенство при этом сохранится, так как осталось прежним или уменьшилось, а увеличилось. не может увеличится, так как функция заменяет последние подряд идущие единицы числа на нули, а по формуле у нового значения увеличивается количество единиц в конце, что не может привести к увеличению . Покажем, почему не стоит рассматривать значения , отличные от тех, которые мы получили по формуле. Рассмотрим такое число , которое больше . Пусть . Уберём у все подряд идущие единицы в конце двоичной записи, столько же цифр уберём в конце числа . Обозначим их как и . Чтобы выполнялось условие , должно выполняться неравенство . Но если , то и , что противоречит условию . Значит, . Но тогда возможно получить по формуле . Получили противоречие, следовательно, . Таким образом, нужные элементы можно искать по формуле . |
Заметим, что возрастает немонотонно. Поэтому нельзя просто перебирать значения от , пока не нарушается условие. Например, при на следующем шаге (), нарушится условие и мы прекратим пересчитывать . Но тогда мы упускаем остальные значения , например .
| , десятичная запись | |||||||||||
| , двоичная запись | |||||||||||
| , двоичная запись | |||||||||||
| , десятичная запись |
Все мы можем получить следующим образом: . Следующим элементом в последовательности будет элемент, у которого первый с конца ноль превратится в единицу. Можно заметить, что если к исходному элементу прибавить единицу, то необходимый ноль обратится в единицу, но при этом все следующие единицы обнулятся. Чтобы обратно их превратить в единицы, применим операцию . Таким образом все нули в конце превратятся в единицы и мы получим нужный элемент. Для того, чтобы понять, что эта последовательность верна, достаточно посмотреть на таблицу.
Несложно заметить, что данная последовательность строго возрастает и в худшем случае будет применена логарифм раз, так как добавляет каждый раз по одной единице в двоичном разложении числа . Напишем функцию, которая будет изменять элемент на , и при этом меняет соответствующие частичные суммы.
function modify(i, d):
while i < N
t[i] += d
i = i | (i + 1)
Запрос получения значения функции на префиксе
Пусть существует некоторая бинарная операция . Чтобы получить значение на отрезке , нужно провести операцию, обратную к , над значениями на отрезках и .
В качестве бинарной операции рассмотрим операцию сложения.
Обозначим . Тогда .
| Лемма: |
входит в сумму для , если . |
Для доказательства леммы рассмотрим битовую запись следующих чисел:
Реализация
Приведем код функции :
int sum(i):
result = 0;
while i >= 0
result += t[i]
i = f(i) - 1
return result
Сравнение дерева Фенвика и дерева отрезков
- И дерево Фенвика, и дерево отрезков занимает памяти, но в случае, когда , дерево Фенвика занимает в 2 раза меньше памяти, чем дерево отрезков.
- Дерево Фенвика проще в реализации.
- Функция, для которой строится дерево Фенвика, должна быть обратимой, а это значит, что минимум и максимум это дерево считать не может (за исключением случаев, когда некоторыми данными мы можем пожертвовать), в отличие от дерева отрезков.
