Дерево Фенвика — различия между версиями
| Строка 7: | Строка 7: | ||
Пусть дан массив <tex> A = [a_0, a_1, ... , a_{n - 1}]</tex>. | Пусть дан массив <tex> A = [a_0, a_1, ... , a_{n - 1}]</tex>. | ||
| − | Деревом Фенвика будем называть массив <tex> T </tex> из <tex> n </tex> элементов: <tex> T_i = \sum\limits_{k = F(i)}^{i} a_k</tex>, где <tex> i = 0 .. n - 1 </tex> и <tex> F(i) </tex> — некоторая функция, от выбора которой зависит время работы операций над деревом. Рассмотрим функцию, позволяющую делать операции вставки и изменения элемента за время <tex> O(\log n) </tex>. Она задается простой формулой: <tex> F(i) = i \And (i + 1) </tex>, где <tex> \And </tex> — это операция логического <tex>AND</tex>. При <tex>AND</tex> числа и его значения, увеличенного на единицу, мы получаем это число без последних подряд идущих единиц. | + | Деревом Фенвика будем называть массив <tex> T </tex> из <tex> n </tex> элементов: <tex> T_i = \sum\limits_{k = F(i)}^{i} a_k</tex>, где <tex> i = 0 .. n - 1 </tex> и <tex> F(i) </tex> — некоторая функция, от выбора которой зависит время работы операций над деревом. Рассмотрим функцию, позволяющую делать операции вставки и изменения элемента за время <tex> O(\log n) </tex>. Она задается простой формулой: <tex> F(i) = i \And (i + 1) </tex>, где <tex> \And </tex> — это операция побитового логического <tex>AND</tex>. При <tex>AND</tex> числа и его значения, увеличенного на единицу, мы получаем это число без последних подряд идущих единиц. |
Эту функцию можно вычислять по другой формуле: <tex> F(i) = i - 2^{h(i)} + 1, </tex> где <tex> h(i) </tex> — количество подряд идущих единиц в конце бинарной записи числа <tex> i </tex>. Оба варианта равносильны, так как функция, заданная какой-либо из этих формул, заменяет все подряд идущие единицы в конце числа на нули. | Эту функцию можно вычислять по другой формуле: <tex> F(i) = i - 2^{h(i)} + 1, </tex> где <tex> h(i) </tex> — количество подряд идущих единиц в конце бинарной записи числа <tex> i </tex>. Оба варианта равносильны, так как функция, заданная какой-либо из этих формул, заменяет все подряд идущие единицы в конце числа на нули. | ||
| Строка 15: | Строка 15: | ||
{{Лемма | {{Лемма | ||
|statement= | |statement= | ||
| − | Для изменения величины <tex>a_{k}</tex> необходимо изменить элементы дерева <tex>T_{i}</tex>, для которых верно неравенство <tex>F(i) \leqslant k \leqslant i</tex> . | + | Для изменения величины <tex>a_{k}</tex> необходимо изменить элементы дерева <tex>T_{i}</tex>, для индексов <tex>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>T_i</tex>, для которых <tex>a_{k}</tex> попадает в <tex>T_i \Rightarrow</tex> необходимые <tex> i </tex> удовлетворяют условию <tex>F(i) \leqslant k \leqslant i</tex>. | <tex> T_i =\sum\limits_{k = F(i)}^{i} a_k , i = 0 .. n - 1 \Rightarrow</tex> необходимо менять те <tex>T_i</tex>, для которых <tex>a_{k}</tex> попадает в <tex>T_i \Rightarrow</tex> необходимые <tex> i </tex> удовлетворяют условию <tex>F(i) \leqslant k \leqslant i</tex>. | ||
| Строка 21: | Строка 21: | ||
{{Лемма | {{Лемма | ||
| − | |statement= Все такие <tex> i </tex> | + | |statement= Все такие <tex> i </tex> можно найти по формуле <tex>i_{next} = i_{prev} \mid (i_{prev} + 1) </tex>, где <tex> \mid </tex> — это операция побитового логического <tex> OR </tex>. |
| − | |proof= | + | |proof=Из доказанной выше леммы следует, что первый элемент последовательности само <tex> k </tex>. Для него выполняется равенство, так как <tex> F(i) \leqslant i </tex>. По формуле <tex>i_{next} = i_{prev} \mid (i_{prev} + 1) </tex> мы заменим первый ноль на единицу. Неравенство при этом сохранится, так как <tex>F(i)</tex> осталось прежним или уменьшилось, а <tex> i </tex> увеличилось. Можем заметить, что если количество единиц в конце не будет совпадать с <tex> k </tex>, то формула <tex>i_{next} = i_{prev} \mid (i_{prev} + 1) </tex> нарушит неравенство, потому что либо само <tex> i </tex> будет меньше, чем <tex>k</tex>, либо <tex> F(i) </tex> станет больше, чем <tex> k </tex>. Таким образом, перебраны будут только нужные элементы}} |
| − | + | ||
| + | Заметим, что <tex>F(i)</tex> возрастает немонотонно. | ||
| + | |||
| + | {| style="background-color:#CCC;margin:0.5px" | ||
| + | |style="background-color:#EEE;padding:2px 30px"| <tex>i</tex>, десятичная запись | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>1</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>2</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>3</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>4</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>5</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>6</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>7</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>8</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>9</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>10</tex> | ||
| + | |||
| + | |- | ||
| + | |style="background-color:#EEE;padding:2px 30px"| <tex>i</tex>, двоичная запись | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0000</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0001</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0010</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0011</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0100</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0101</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0110</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0111</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>1000</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>1001</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>1010</tex> | ||
| + | |- | ||
| + | |style="background-color:#EEE;padding:2px 30px"| <tex>F(i)</tex>, двоичная запись | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0000</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0000</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0010</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0000</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0100</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0100</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0110</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0000</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>1000</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>1000</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>1010</tex> | ||
| + | |- | ||
| + | |style="background-color:#EEE;padding:2px 30px"| <tex>F(i)</tex>, десятичная запись | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>2</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>4</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>4</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>6</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>0</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>8</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>8</tex> | ||
| + | |style="background-color:#FFF;padding:2px 30px"| <tex>10</tex> | ||
| + | |} | ||
| + | |||
| + | |||
Все <tex>i</tex> мы можем получить следующим образом : <tex>i_{next} = i_{prev} \mid (i_{prev} + 1) </tex>. Следующим элементом в последовательности будет элемент, у которого первый с конца ноль превратится в единицу. Можно заметить, что если к исходному элементу прибавить единицу, то необходимый ноль обратится в единицу, но при этом все следующие единицы обнулятся. Чтобы обратно их превратить в единицы, применим операцию <tex>OR</tex>. Таким образом все нули в конце превратятся в единицы и мы получим нужный элемент. Для того, чтобы понять, что эта последовательность верна, достаточно посмотреть на таблицу. | Все <tex>i</tex> мы можем получить следующим образом : <tex>i_{next} = i_{prev} \mid (i_{prev} + 1) </tex>. Следующим элементом в последовательности будет элемент, у которого первый с конца ноль превратится в единицу. Можно заметить, что если к исходному элементу прибавить единицу, то необходимый ноль обратится в единицу, но при этом все следующие единицы обнулятся. Чтобы обратно их превратить в единицы, применим операцию <tex>OR</tex>. Таким образом все нули в конце превратятся в единицы и мы получим нужный элемент. Для того, чтобы понять, что эта последовательность верна, достаточно посмотреть на таблицу. | ||
Версия 15:18, 27 мая 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
Преимущества и недостатки дерева Фенвика
Главными преимуществами данной конструкции являются простота реализации и быстрота ответов на запросы за . Также дерево Фенвика позволяет быстро изменять значения в массиве и находить некоторые функции от элементов массива. Недостатком является то, что при изменении одного элемента исходного массива, приходится пересчитывать частичные суммы, а это затратно по времени.
