Левосторонняя куча — различия между версиями
Shersh (обсуждение | вклад) (→decreaseKey) |
Shersh (обсуждение | вклад) |
||
Строка 3: | Строка 3: | ||
Левосторонние деревья были изобретены Кларком Крейном (Clark Allan Crane), свое название они получили из-за того, что левое поддерево обычно длиннее правого. | Левосторонние деревья были изобретены Кларком Крейном (Clark Allan Crane), свое название они получили из-за того, что левое поддерево обычно длиннее правого. | ||
{{Определение | {{Определение | ||
− | |definition='''Левосторонняя куча (leftist heap | + | |definition='''Левосторонняя куча''' (англ. ''leftist heap'') {{---}} двоичное левосторонее [[Дерево, эквивалентные определения|дерево]] (не обязательно сбалансированное), но с соблюдением [[Двоичная куча#Определение|порядка кучи]] (heap order).}} |
'''Свободной позицией''' назовем место в дереве, куда может быть вставлена новая вершина. Само дерево будет являться свободной позицией, если оно не содержит вершин. Если же у какой-то внутренней вершины нет сына, то на его месте {{---}} ''свободная позиция''. | '''Свободной позицией''' назовем место в дереве, куда может быть вставлена новая вершина. Само дерево будет являться свободной позицией, если оно не содержит вершин. Если же у какой-то внутренней вершины нет сына, то на его месте {{---}} ''свободная позиция''. | ||
{{Лемма | {{Лемма | ||
Строка 15: | Строка 15: | ||
{{Определение | {{Определение | ||
− | |definition='''Условие левосторонней кучи'''. Пусть <tex>dist(u)</tex> {{---}} расстояние от вершины <tex>u</tex> до ближайшей свободной позиции в ее поддереве. У пустых позиций <tex>dist = 0</tex>. Тогда потребуем для любой вершины <tex>u : dist(u. | + | |definition='''Условие левосторонней кучи'''. Пусть <tex>dist(u)</tex> {{---}} расстояние от вершины <tex>u</tex> до ближайшей свободной позиции в ее поддереве. У пустых позиций <tex>dist = 0</tex>. Тогда потребуем для любой вершины <tex>u : dist(u.left)\geqslant dict(u.right)</tex>.}} |
Если для какой- то вершины это свойство не выполняется, то это легко устраняется: можно за <tex>O(1)</tex> поменять местами левого и правого ребенка, что не повлияет на порядок кучи. | Если для какой- то вершины это свойство не выполняется, то это легко устраняется: можно за <tex>O(1)</tex> поменять местами левого и правого ребенка, что не повлияет на порядок кучи. | ||
Строка 23: | Строка 23: | ||
Слияние двух куч. | Слияние двух куч. | ||
− | ''' | + | '''LeftistHeap''' merge(x, y): <font color=darkgreen>// x, y {{---}} корни двух деревьев</font> |
− | '''if''' x == <tex> \varnothing </tex>: '''return''' y | + | '''if''' x == <tex> \varnothing </tex>: |
− | '''if''' y == <tex> \varnothing </tex>: '''return''' x | + | '''return''' y |
+ | '''if''' y == <tex> \varnothing </tex>: | ||
+ | '''return''' x | ||
'''if''' y.key < x.key: | '''if''' y.key < x.key: | ||
swap(x, y) | swap(x, y) | ||
− | // Воспользуемся тем, что куча левосторонняя. Правая ветка {{---}} самая короткая и не длиннее логарифма. | + | <font color=darkgreen>// Воспользуемся тем, что куча левосторонняя. Правая ветка {{---}} самая короткая и не длиннее логарифма. |
− | // Пойдем направо и сольем правое поддерево с у. | + | // Пойдем направо и сольем правое поддерево с у.</font> |
− | x. | + | x.right = '''merge'''(x.right, y) |
− | // Могло возникнуть нарушение левостороннести кучи | + | <font color=darkgreen>// Могло возникнуть нарушение левостороннести кучи</font> |
− | '''if''' dist(x. | + | '''if''' dist(x.right) > dist(x.left): |
− | swap(x. | + | swap(x.left, x.right) |
− | dist(x) = min(dist(x. | + | dist(x) = min(dist(x.left), dist(x.right)) + 1 <font color=darkgreen>// пересчитаем расстояние до ближайшей свободной позиции</font> |
'''return''' x | '''return''' x | ||
− | // Каждый раз идем из уже существующей вершины только в правое поддерево {{---}} не более логарифма вызовов (по лемме) | + | <font color=darkgreen>// Каждый раз идем из уже существующей вершины только в правое поддерево {{---}} не более логарифма вызовов (по лемме)</font> |
Левосторонняя куча относится к сливаемым кучам: остальные операции легко реализуются с помощью операции слияния. | Левосторонняя куча относится к сливаемым кучам: остальные операции легко реализуются с помощью операции слияния. | ||
Строка 45: | Строка 47: | ||
Как и у любой другой двоичной кучи, минимум хранится в корне. Извлекаем минимальное значение, удаляем корень, сливаем левое и правое поддерево корня. Возвращает пару из извлеченной вершины и новой кучи. | Как и у любой другой двоичной кучи, минимум хранится в корне. Извлекаем минимальное значение, удаляем корень, сливаем левое и правое поддерево корня. Возвращает пару из извлеченной вершины и новой кучи. | ||
===delete=== | ===delete=== | ||
− | Аналогично удаляется любой элемент {{---}} на его место ставится результат слияния его детей. Но так просто любой элемент удалить нельзя {{---}} на пути от этого элемента к корню может нарушиться левостороннесть кучи. А до корня мы дойти не можем, так как элемент может находиться на линейной глубине. Поэтому удаление реализуется с помощью <tex> | + | Аналогично удаляется любой элемент {{---}} на его место ставится результат слияния его детей. Но так просто любой элемент удалить нельзя {{---}} на пути от этого элемента к корню может нарушиться левостороннесть кучи. А до корня мы дойти не можем, так как элемент может находиться на линейной глубине. Поэтому удаление реализуется с помощью <tex>\mathrm{decreaseKey}</tex>. Уменьшаем ключ до <tex>-\infty</tex>, затем извлекаем минимальное значение. |
===decreaseKey=== | ===decreaseKey=== | ||
{{Лемма | {{Лемма | ||
Строка 66: | Строка 68: | ||
|about=3 | |about=3 | ||
|statement= Нужно транспонировать не более <tex>\log{n}</tex> поддеревьев. | |statement= Нужно транспонировать не более <tex>\log{n}</tex> поддеревьев. | ||
− | |proof=Длина пути от вершины до корня может быть и <tex>O(n)</tex>, но нам не нужно подниматься до корня {{---}} достаточно подняться до вершины, у которой свойство левосторонней кучи уже выполнено. Транспонируем только если <tex>dist(x. | + | |proof=Длина пути от вершины до корня может быть и <tex>O(n)</tex>, но нам не нужно подниматься до корня {{---}} достаточно подняться до вершины, у которой свойство левосторонней кучи уже выполнено. Транспонируем только если <tex>dist(x.left) < dist(x.right)</tex>, но <tex>dist(x.right) \leqslant \log{n}</tex>. На каждом шаге, если нужно транспонируем и увеличиваем <tex>dist</tex>, тогда <tex>dist</tex> увеличится до <tex>\log{n}</tex> и обменов уже не надо будет делать.}} |
− | Таким образом, мы восстановили левостороннесть кучи за <tex>O(\log{n})</tex>. Поэтому асимптотика операции <tex> decreaseKey </tex> {{---}} <tex>O(\log{n})</tex>. | + | Таким образом, мы восстановили левостороннесть кучи за <tex>O(\log{n})</tex>. Поэтому асимптотика операции <tex>\mathrm{decreaseKey} </tex> {{---}} <tex>O(\log{n})</tex>. |
==Построение кучи за O(n)== | ==Построение кучи за O(n)== | ||
Строка 95: | Строка 97: | ||
После подстановки <tex> x = \genfrac{}{}{}{0}{1}{2} </tex> получаем, что сумма равна <tex> 2 </tex>. Следовательно, построение кучи таким образом произойдёт за <tex> O(n) </tex>. | После подстановки <tex> x = \genfrac{}{}{}{0}{1}{2} </tex> получаем, что сумма равна <tex> 2 </tex>. Следовательно, построение кучи таким образом произойдёт за <tex> O(n) </tex>. | ||
==Преимущества левосторонней кучи== | ==Преимущества левосторонней кучи== | ||
− | Нигде не делается уничтожающих присваиваний. Не создается новых узлов в <tex>merge</tex>. Эта реализация слияния является функциональной — ее легко реализовать на функциональном языке программирования. Также данная реализация <tex>merge</tex> является персистентной. | + | Нигде не делается уничтожающих присваиваний. Не создается новых узлов в <tex>\mathrm{merge}</tex>. Эта реализация слияния является функциональной — ее легко реализовать на функциональном языке программирования. Также данная реализация <tex>\mathrm{merge}</tex> является персистентной. |
− | == | + | ==Источники информации== |
− | + | * [http://compscicenter.ru/program/lecture/6829 Лекция "Приоритетные очереди" А. С. Станкевича в Computer Science Center] | |
+ | * [http://www.intuit.ru/studies/courses/100/100/lecture/1539?page=1 Левосторонние кучи. Интуит.] | ||
+ | * [[wikipedia:Leftist_tree|Wikipedia {{---}} Leftist tree]] | ||
− | |||
− | |||
− | |||
[[Категория: Дискретная математика и алгоритмы]] | [[Категория: Дискретная математика и алгоритмы]] | ||
[[Категория: Приоритетные очереди]] | [[Категория: Приоритетные очереди]] |
Версия 16:52, 1 сентября 2014
Содержание
Условие кучи
Левосторонние деревья были изобретены Кларком Крейном (Clark Allan Crane), свое название они получили из-за того, что левое поддерево обычно длиннее правого.
Определение: |
Левосторонняя куча (англ. leftist heap) — двоичное левосторонее дерево (не обязательно сбалансированное), но с соблюдением порядка кучи (heap order). |
Свободной позицией назовем место в дереве, куда может быть вставлена новая вершина. Само дерево будет являться свободной позицией, если оно не содержит вершин. Если же у какой-то внутренней вершины нет сына, то на его месте — свободная позиция.
Лемма (1): |
В двоичном дереве с вершинами существует свободная позиция на глубине не более . |
Доказательство: |
Если бы все свободные позиции были на глубине более логарифма, то мы получили бы полное дерево с количеством вершин более | .
Левосторонняя куча накладывает на двоичное дерево дополнительное условие. Ближайшая свободная позиция должна быть самой правой позицией в дереве. То есть помимо обычного условия кучи выполняется следующее:
Определение: |
Условие левосторонней кучи. Пусть | — расстояние от вершины до ближайшей свободной позиции в ее поддереве. У пустых позиций . Тогда потребуем для любой вершины .
Если для какой- то вершины это свойство не выполняется, то это легко устраняется: можно за поменять местами левого и правого ребенка, что не повлияет на порядок кучи.
Поддерживаемые операции
merge
Слияние двух куч.
LeftistHeap merge(x, y): // x, y — корни двух деревьев if x ==: return y if y == : return x if y.key < x.key: swap(x, y) // Воспользуемся тем, что куча левосторонняя. Правая ветка — самая короткая и не длиннее логарифма. // Пойдем направо и сольем правое поддерево с у. x.right = merge(x.right, y) // Могло возникнуть нарушение левостороннести кучи if dist(x.right) > dist(x.left): swap(x.left, x.right) dist(x) = min(dist(x.left), dist(x.right)) + 1 // пересчитаем расстояние до ближайшей свободной позиции return x // Каждый раз идем из уже существующей вершины только в правое поддерево — не более логарифма вызовов (по лемме)
Левосторонняя куча относится к сливаемым кучам: остальные операции легко реализуются с помощью операции слияния.
insert
Вставка новой вершины в дерево. Новое левостороннее дерево, состоящее из одной вершины, сливается с исходным.
extractMin
Как и у любой другой двоичной кучи, минимум хранится в корне. Извлекаем минимальное значение, удаляем корень, сливаем левое и правое поддерево корня. Возвращает пару из извлеченной вершины и новой кучи.
delete
Аналогично удаляется любой элемент — на его место ставится результат слияния его детей. Но так просто любой элемент удалить нельзя — на пути от этого элемента к корню может нарушиться левостороннесть кучи. А до корня мы дойти не можем, так как элемент может находиться на линейной глубине. Поэтому удаление реализуется с помощью
. Уменьшаем ключ до , затем извлекаем минимальное значение.decreaseKey
Лемма (2): |
У левостороннего дерева с правой ветвью длинны количество узлов . |
Доказательство: |
Индукция по h. При — верно.При По индукции число узлов в каждом из них больше или равно левое и правое поддеревья исходного дерева левосторонние, а от их корней больше либо равен . , тогда во всем дереве узлов. |
Алгоритм
- Найдем узел , вырежем поддерево с корнем в этом узле.
- Пройдем от предка вырезанной вершины, при этом пересчитывая . Если левого сына вершины меньше правого, то меняем местами поддеревья.
- Уменьшаем ключ данного узла и сливаем два дерева: исходное и вырезанное.
Лемма (3): |
Нужно транспонировать не более поддеревьев. |
Доказательство: |
Длина пути от вершины до корня может быть и | , но нам не нужно подниматься до корня — достаточно подняться до вершины, у которой свойство левосторонней кучи уже выполнено. Транспонируем только если , но . На каждом шаге, если нужно транспонируем и увеличиваем , тогда увеличится до и обменов уже не надо будет делать.
Таким образом, мы восстановили левостороннесть кучи за
. Поэтому асимптотика операции — .Построение кучи за O(n)
Храним список левосторонних куч. Пока их количество больше
, из начала списка достаем две кучи, сливаем их и кладем в конец списка.Покажем, почему это будет работать за
.Пусть на
-ом шаге алгоритма в нашем списке остались только кучи размера . На нулевом шаге у нас куч из одного элемента, и на каждом следующем количество куч будет уменьшаться вдвое, а число вершин в куче будет увеличиваться вдвое. Слияние двух куч из элементов выполняется за . Поэтому построение будет выполняться за
Покажем, что сумма — ряда, заменив его на эквивалентный функциональный:
, тогда и алгоритм будет выполняться за . Найдём сумму
После подстановки
получаем, что сумма равна . Следовательно, построение кучи таким образом произойдёт за .Преимущества левосторонней кучи
Нигде не делается уничтожающих присваиваний. Не создается новых узлов в
. Эта реализация слияния является функциональной — ее легко реализовать на функциональном языке программирования. Также данная реализация является персистентной.