Фибоначчиева куча — различия между версиями
| Строка 2: | Строка 2: | ||
{{Определение | {{Определение | ||
|definition= | |definition= | ||
| − | '''Фибоначчиево дерево''' (англ. | + | '''Фибоначчиево дерево''' (англ. ''Fibonacci tree'') {{---}} [[Биномиальная куча#Биномиальное дерево |биномиальное дерево]], где у каждой вершины удалено не более одного ребенка. |
}} | }} | ||
{{Определение | {{Определение | ||
|definition= | |definition= | ||
| − | '''Порядок фибоначчиева дерева''' (англ. | + | '''Порядок фибоначчиева дерева''' (англ. ''Fibonacci tree order'') {{---}} порядок соответствующего биномиального дерева, из которого оно получено. |
}} | }} | ||
{{Определение | {{Определение | ||
|definition= | |definition= | ||
| − | '''Степень вершины''' (англ. | + | '''Степень вершины''' (англ. ''degree'') {{---}} количество дочерних узлов данной вершины. |
}} | }} | ||
| Строка 67: | Строка 67: | ||
{{Определение | {{Определение | ||
|definition= | |definition= | ||
| − | '''Фибоначчиева куча''' (англ. | + | '''Фибоначчиева куча''' (англ. ''Fibonacci heap'') {{---}} набор фибоначчиевых деревьев, корни которых объединены в неупорядоченный [[Список#Циклический список |циклический]] [[Список#Двусвязный список | двусвязный список]]. В отличие от [[Биномиальная куча|биномиальной кучи]], степени корней не обязаны быть попарно различными. |
}} | }} | ||
Версия 20:19, 20 мая 2015
Содержание
Фибоначчиево дерево
| Определение: |
| Фибоначчиево дерево (англ. Fibonacci tree) — биномиальное дерево, где у каждой вершины удалено не более одного ребенка. |
| Определение: |
| Порядок фибоначчиева дерева (англ. Fibonacci tree order) — порядок соответствующего биномиального дерева, из которого оно получено. |
| Определение: |
| Степень вершины (англ. degree) — количество дочерних узлов данной вершины. |
| Лемма: |
Для всех целых
, где — -ое число Фибоначчи, определяемое формулой: |
| Доказательство: |
|
Докажем лемму по индукции: при , что действительно верно. По индукции предполагаем, что . Тогда |
| Лемма: |
Фибоначчиево дерево порядка содержит не менее вершин. |
| Доказательство: |
|
Докажем это утверждение по индукции. Пусть — минимальный размер фибоначчиева дерева порядка . При . При . Предположим по индукции, что для всех . Пусть в нашем дереве удалено поддерево порядка . Тогда
Но по предыдущей лемме : . Следовательно, |
Фибоначчиева куча
| Определение: |
| Фибоначчиева куча (англ. Fibonacci heap) — набор фибоначчиевых деревьев, корни которых объединены в неупорядоченный циклический двусвязный список. В отличие от биномиальной кучи, степени корней не обязаны быть попарно различными. |
Фибоначчиевы кучи поддерживают тот же набор операций, что и биномиальные кучи, но имеют то преимущество, что операции, в которых не требуется удаление, имеют амортизированное время работы, равное .
С теоретической точки зрения фибоначчиевы кучи особенно полезны в случае, когда количество операций и относительно мало по сравнению с количеством других операций. Однако с практической точки зрения программная сложность и высокие значения постоянных множителей в формулах времени работы существенно снижают эффективность применения фибоначчиевых куч, делая их в большинстве случаев менее привлекательными, чем обычные бинарные кучи.
| Лемма: |
, где |
| Доказательство: |
|
Для начала докажем, что Используем для этого математическую индукцию. При , что верно. При , что также верно. По индукции предполагаем, что и . Тогда
Подставив вместо его значение, нетрудно убедится, что Поскольку , то выполняются неравенства . Таким образом, -ое число Фибоначчи равно , округленному до ближайшего целого числа. Следовательно, . |
| Лемма: |
Максимальная степень произвольной вершины в фибоначчиевой куче с вершинами равна |
| Доказательство: |
|
Пусть — произвольная вершина в фибоначчиевой куче с вершинами, и пусть — степень вершины . Тогда по доказанному выше в дереве, корень которого , содержится не менее вершин, что в свою очередь по лемме равно . То есть
Логарифмируя по основанию , получаем Таким образом, максимальная степень произвольной вершины равна . |
Структура
- Каждый узел в куче содержит следующие указатели и поля:
Node int key; //ключ Node* p; //указатель на родительский узел Node* child; //указатель на один из дочерних узлов Node* left; //указатель на левый сестринский узел Node* right; //указатель на правый сестринский узел int degree; //количество дочерних узлов boolean mark; //флаг, который показывает, удаляли ли мы дочерние узлы данной вершины
- Дочерние узлы объединены при помощи указателей и в циклический двусвязный список.
- Корни всех деревьев в связаны при помощи указателей и в циклический двусвязный список корней.
- Обращение к выполняется посредством указателя на корень дерева с минимальным ключом. Этот узел называется минимальным узлом .
- Текущее количество узлов в хранится в .
Циклический двусвязный список обладает двумя преимуществами для использования в фибоначчиевых кучах. Во-первых, удаление элемента из такого списка выполняется за время . Во-вторых, если имеется два таких списка, их легко объединить в один за время .
Потенциал
Для анализа производительности операций введем потенциал для фибоначчиевой кучи как , где — количество элементов в корневом списке кучи, а — количество вершин, у которых удален один ребенок (то есть вершин с пометкой ). Договоримся, что единицы потенциала достаточно для оплаты константного количества работы.
Операции
Рассмотрим операции, которые поддерживают фибоначчиевы кучи. Амортизированное время их работы показано в таблице.
| Операция | Амортизированная сложность |
|---|---|
Стоит заметить, что структура фибоначчиевых куч, также как биномиальных и бинарных, не могут обеспечить эффективную реализацию поиска элемента с заданным ключом, поэтому операции и получают в качестве аргумента указатель на узел, а не значение его ключа.
makeHeap
Создается новый пустой корневой список, в устанавливается значение . Реальное время работы — .
insert
Вставка элемента в фибоначчиеву кучу также тривиальна: создается новая куча из одного элемента и сливается с текущей. Для оценки амортизированной стоимости операции рассмотрим исходную кучу и получившуюся в результате вставки нового элемента кучу . и . Следовательно, увеличение потенциала составляет . Так как реальное время работы составляет , то амортизированная стоимость данной операции также равна .
getMin
Возвращает указатель . Реальное время работы — .
merge
Слияние двух фибоначчиевых куч происходит просто: объединяем списки этих куч в один, релаксируем минимум. Реальное время работы — . Амортизированное время работы также , поскольку, при объединении двух куч в одну, потенциалы обеих куч суммируются, итоговая сумма потенциалов не изменяется, .
extractMin
Первая рассматриваемая операция, в ходе которой меняется структура кучи. Здесь используется вспомогательная процедура . Возьмем указатель на , удалим эту вершину. Ее поддеревья (их не более, чем , где — максимальная степень вершины в куче) объединим с корневым списком. Теперь вызываем процедуру . После этой операции в списке корней остается не более чем узлов, среди которых нужно найти минимальный. Итоговая асимптотика операции , учитывая и вспомогательную функцию , время работы которой доказывается ниже, равно: . По доказанной выше лемме .
consolidate
Данная процедура принимает кучу и преобразует ее таким образом, что в корневом списке остается не более вершин.
Для этого возьмем массив списков указателей на корни деревьев , где — максимальная степень вершины в текущем корневом списке.
Затем происходит процесс, аналогичный слиянию биномиальных куч: добавляем поочередно каждый корень, смотря на его степень. Пусть она равна . Если в соответствующей ячейке еще нету вершины, записываем текущую вершину туда. Иначе подвешиваем одно дерево к другому, и пытаемся также добавить дерево, степень корня которого уже равна . Продолжаем, пока не найдем свободную ячейку.
Учетная стоимость равна . Докажем это:
Изначально в корневом списке было не более вершин, поскольку он состоит из исходного списка корней с узлами, минус извлеченный узел и плюс дочерние узлы, количество которых не превышает . В ходе операции мы сделали слияний деревьев. Потенциал перед извлечением минимума равен , а после не превышает , поскольку в корневом списке остается не более узлов, а количество помеченных узлов не изменяется. Таким образом, амортизированная стоимость не превосходит
Поскольку мы договорились, что можем масштабировать единицу потенциала таким образом, чтобы покрывать константное количество работы, то итоговая амортизационная оценка —
decreaseKey
Основная идея: хотим, чтобы учетная стоимость данной операции была . Было бы хорошо, чтобы вершина не всплывала до корня, и тогда дерево не придется сильно перестраивать. Для этого при удобном случае будем вырезать поддерево полностью и перемещать его в корневой список. Итак, сам алгоритм:
- Проверяем, если новое значение ключа все же не меньше значения ключа родителя, то все хорошо, и мы выходим.
- Иначе, вырезаем дерево с текущей вершиной в корневой список, и производим каскадное вырезание родителя.
cut
При вырезании вершины мы удаляем ее из списка детей своего родителя, уменьшаем степень ее родителя () и снимаем пометку с текущей вершины ().
cascadingCut
Перед вызовом каскадного вырезания нам известно, удаляли ли ребенка у этой вершины. Если у вершины до этого не удаляли дочерний узел (), то мы помечаем эту вершину () и прекращаем выполнение операции. В противном случае применяем операцию для текущей вершины и запускаем каскадное вырезание от родителя.
Пример
Рисунок иллюстрирует пример каскадного вырезания:
- Изначально, куча состояла из фибоначчиевых деревьев. У вершины с ключом отсутствует ребенок.
- Уменьшаем ключ до и делаем операцию этого дерева. Получаем кучу с деревьями и новым минимумом. Но у вершины с ключом был удален второй ребенок, поэтому запускам операцию для этой вершины: вырезаем ее, помещаем в корневой список и помечаем ее родителя.
- У вершины с ключом удален лишь один ребенок, поэтому операция от нее не запускается. В итоге, получаем кучу, состоящую из фибоначчиевых деревьев.
Время работы
Докажем, что амортизированное время работы операции есть . Поскольку в процедуре нет циклов, ее время работы определяется лишь количеством рекурсивных вызовов каскадного вырезания.
Пусть мы вызвали процедуру каскадного вырезания раз. Так как реальное время работы операции без учета рекурсии составляет , то реальное время работы операции — .
Рассмотрим, как изменится потенциал в результате выполнения данной операции. Пусть — фибоначчиева куча до вызова . Тогда после рекурсивных вызовов операции вершин с пометкой стало как минимум на меньше, потому что каждый вызов каскадного вырезания, за исключением последнего, уменьшает количество помеченных вершин на одну, и в результате последнего вызова одну вершину мы можем пометить. В корневом списке прибавилось новых деревьев ( дерево за счет каскадного вырезания и еще одно из-за самого первого вызова операции ).
В итоге, изменение потенциала составляет: . Следовательно, амортизированная стоимость не превышает . Но поскольку мы можем соответствующим образом масштабировать единицы потенциала, то амортизированная стоимость операции равна .
delete
Удаление вершины реализуется через уменьшение ее ключа до и последующим извлечением минимума. Амортизированное время работы: .
Поскольку ранее мы показали, что , то соответствующие оценки доказаны.
Источники информации
- Томас Кормен, Чарльз Лейзерсон, Рональд Ривест, Клиффорд Штайн — Алгоритмы: построение и анализ. — М.: Издательский дом «Вильямс», 2005. — С. 1296. — ISBN 5-8459-0857-4
- Числа Фибоначчи — Википедия
- Фибоначчиева куча — Википедия
- Фибоначчиевы кучи — INTUIT.ru
- Визуализаторы
- Fibonacci Heaps