Левосторонняя куча — различия между версиями
(dist(x) = min(dist(x.right), dist(x.left))+1 исправлено на dist(x) = dist(x.right)+1 (правая ветвь стала меньше на предыдущем шаге)→merge) |
|||
| Строка 1: | Строка 1: | ||
| + | {| class="wikitable" align="center" style="color: red; background-color: black; font-size: 56px; width: 800px;" | ||
| + | |+ | ||
| + | |-align="center" | ||
| + | |'''НЕТ ВОЙНЕ''' | ||
| + | |-style="font-size: 16px;" | ||
| + | | | ||
| + | 24 февраля 2022 года российское руководство во главе с Владимиром Путиным развязало агрессивную войну против Украины. В глазах всего мира это военное преступление совершено от лица всей страны, всех россиян. | ||
| + | |||
| + | Будучи гражданами Российской Федерации, мы против своей воли оказались ответственными за нарушение международного права, военное вторжение и массовую гибель людей. Чудовищность совершенного преступления не оставляет возможности промолчать или ограничиться пассивным несогласием. | ||
| + | |||
| + | Мы убеждены в абсолютной ценности человеческой жизни, в незыблемости прав и свобод личности. Режим Путина — угроза этим ценностям. Наша задача — обьединить все силы для сопротивления ей. | ||
| + | |||
| + | Эту войну начали не россияне, а обезумевший диктатор. И наш гражданский долг — сделать всё, чтобы её остановить. | ||
| + | |||
| + | ''Антивоенный комитет России'' | ||
| + | |-style="font-size: 16px;" | ||
| + | |Распространяйте правду о текущих событиях, оберегайте от пропаганды своих друзей и близких. Изменение общественного восприятия войны - ключ к её завершению. | ||
| + | |-style="font-size: 16px;" | ||
| + | |[https://meduza.io/ meduza.io], [https://www.youtube.com/c/popularpolitics/videos Популярная политика], [https://novayagazeta.ru/ Новая газета], [https://zona.media/ zona.media], [https://www.youtube.com/c/MackNack/videos Майкл Наки]. | ||
| + | |} | ||
| + | |||
[[Файл:LeftistHeap.jpg|400px|thumb|right|Левосторонняя куча]] | [[Файл:LeftistHeap.jpg|400px|thumb|right|Левосторонняя куча]] | ||
==Условие кучи== | ==Условие кучи== | ||
Версия 09:25, 1 сентября 2022
| НЕТ ВОЙНЕ |
|
24 февраля 2022 года российское руководство во главе с Владимиром Путиным развязало агрессивную войну против Украины. В глазах всего мира это военное преступление совершено от лица всей страны, всех россиян. Будучи гражданами Российской Федерации, мы против своей воли оказались ответственными за нарушение международного права, военное вторжение и массовую гибель людей. Чудовищность совершенного преступления не оставляет возможности промолчать или ограничиться пассивным несогласием. Мы убеждены в абсолютной ценности человеческой жизни, в незыблемости прав и свобод личности. Режим Путина — угроза этим ценностям. Наша задача — обьединить все силы для сопротивления ей. Эту войну начали не россияне, а обезумевший диктатор. И наш гражданский долг — сделать всё, чтобы её остановить. Антивоенный комитет России |
| Распространяйте правду о текущих событиях, оберегайте от пропаганды своих друзей и близких. Изменение общественного восприятия войны - ключ к её завершению. |
| meduza.io, Популярная политика, Новая газета, zona.media, Майкл Наки. |
Содержание
Условие кучи
Левосторонние деревья были изобретены Кларком Крейном (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) = dist(x.right) + 1 // пересчитаем расстояние до ближайшей свободной позиции return x // Каждый раз идем из уже существующей вершины только в правое поддерево — не более логарифма вызовов (по лемме)
Левосторонняя куча относится к сливаемым кучам: остальные операции легко реализуются с помощью операции слияния.
insert
Вставка новой вершины в дерево. Новое левостороннее дерево, состоящее из одной вершины, сливается с исходным.
extractMin
Как и у любой другой двоичной кучи, минимум хранится в корне. Извлекаем минимальное значение, удаляем корень, сливаем левое и правое поддерево корня. Возвращает пару из извлеченной вершины и новой кучи.
delete
Аналогично удаляется любой элемент — на его место ставится результат слияния его детей. Но так просто любой элемент удалить нельзя — на пути от этого элемента к корню может нарушиться левостороннесть кучи. А до корня мы дойти не можем, так как элемент может находиться на линейной глубине. Поэтому удаление реализуется с помощью . Уменьшаем ключ до , затем извлекаем минимальное значение.
decreaseKey
| Лемма (2): |
У левостороннего дерева с правой ветвью длины количество узлов . |
| Доказательство: |
|
Индукция по h. При — верно. При левое и правое поддеревья исходного дерева левосторонние, а от их корней больше либо равен . По индукции число узлов в каждом из них больше или равно , тогда во всем дереве узлов. |
Алгоритм
- Найдем узел , вырежем поддерево с корнем в этом узле.
- Пройдем от предка вырезанной вершины, при этом пересчитывая . Если левого сына вершины меньше правого, то меняем местами поддеревья.
- Уменьшаем ключ данного узла и сливаем два дерева: исходное и вырезанное.
| Лемма (3): |
Нужно транспонировать не более поддеревьев. |
| Доказательство: |
| Длина пути от вершины до корня может быть и , но нам не нужно подниматься до корня — достаточно подняться до вершины, у которой свойство левосторонней кучи уже выполнено. Транспонируем только если , но . На каждом шаге, если нужно транспонируем и увеличиваем , тогда увеличится до и обменов уже не надо будет делать. |
Таким образом, мы восстановили левостороннесть кучи за . Поэтому асимптотика операции — .
Построение кучи за O(n)
Храним список левосторонних куч. Пока их количество больше , из начала списка достаем две кучи, сливаем их и кладем в конец списка.
Покажем, почему это будет работать за .
Пусть на -ом шаге алгоритма в нашем списке остались только кучи размера . На нулевом шаге у нас куч из одного элемента, и на каждом следующем количество куч будет уменьшаться вдвое, а число вершин в куче будет увеличиваться вдвое. Слияние двух куч из элементов выполняется за . Поэтому построение будет выполняться за
Покажем, что сумма — , тогда и алгоритм будет выполняться за . Найдём сумму ряда, заменив его на эквивалентный функциональный:
После подстановки получаем, что сумма равна . Следовательно, построение кучи таким образом произойдёт за .
Преимущества левосторонней кучи
Нигде не делается уничтожающих присваиваний. Не создается новых узлов в . Эта реализация слияния является функциональной — ее легко реализовать на функциональном языке программирования. Также данная реализация является персистентной.
