WikiDer > B-куча

B-heap

А B-куча это двоичная куча реализовано для хранения поддеревьев в одном страница. Это уменьшает количество страниц, к которым осуществляется доступ, до десяти раз для больших куч при использовании виртуальная память, по сравнению с традиционной реализацией.[1] Традиционное сопоставление элементов с местоположениями в множество помещает почти каждый уровень на отдельную страницу.

Существуют и другие варианты кучи, которые эффективны на компьютерах, использующих виртуальную память или кеши, например алгоритмы без кеширования, k-груды,[2] и макеты ван Эмде Боаса.[3]

Мотивация

Традиционно бинарные деревья размещаются в последовательной памяти в соответствии с п -> {2n, 2n + 1} правило, означающее, что если узел находится в позиции п, его левый и правый дочерние элементы считаются находящимися на позициях 2n и 2n + 1 в массиве. Корень находится в позиции 1. Обычной операцией с бинарными деревьями является вертикальный обход; переход по уровням дерева вниз, чтобы добраться до искомого узла. Однако из-за того, как память на современных компьютерах организована в виде страниц в виртуальной памяти, такая схема построения двоичного дерева может быть крайне неэффективной. Причина в том, что, как и при углублении в дерево, расстояние до следующего узла растет экспоненциально, поэтому каждый следующий извлеченный узел, вероятно, будет на отдельной странице памяти. Это увеличит количество пропуск страницыB-куча решает эту проблему, размещая дочерние узлы в памяти другим способом, пытаясь как можно больше позиционировать поддеревья на одной странице. Следовательно, по мере вертикального обхода несколько последовательных извлекаемых узлов будут лежать на одной странице, что приведет к небольшому количеству пропусков страниц.

Выполнение

Подробно b-heap может быть реализован следующим образом. Поул-Хеннинг Камп[4] дает два варианта компоновки узлов: один, при котором две позиции на странице теряются, но строгая двоичная структура дерева сохраняется, и другой, который использует все доступное пространство страниц, но дерево не может расширяться для одного уровня при входе на новую страницу (узлы на этом уровне имеют только одного дочернего элемента). В любом случае важным моментом является то, что при выходе со страницы оба дочерних узла всегда находятся на общей другой странице, поскольку при вертикальном переходе алгоритм обычно сравнивает обоих дочерних узлов с родительским, чтобы знать, как действовать дальше. По этой причине можно сказать, что каждая страница содержит два параллельных поддерева, перемежающихся друг с другом. Сами страницы можно рассматривать как м-арное дерево, и поскольку половина элементов на каждой странице будут листьями (внутри страницы), «дерево страниц» имеет коэффициент разделения размер страницы / 2.

Родительская функция

В отличие от классического макета, подобного массиву, родительская функция в B-куче более сложна, потому что индекс родительского узла должен вычисляться по-разному в зависимости от того, где он находится на странице. Предполагая, что позиции на странице помечены от 0 до размер страницы, родительская функция может быть следующей.

Для узлов 0 и 1 они используются только в варианте, использующем все возможное пространство. В этом случае родительский индекс обоих узлов одинаков, он находится на другой странице, и его конкретное смещение внутри этой страницы зависит только от номера текущей страницы. В частности, чтобы вычислить номер родительской страницы, просто разделите номер страницы текущего узла на коэффициент разделения "дерева страниц", который равен размер страницы / 2. Чтобы получить правильное смещение на странице, учтите, что это должен быть один из конечных узлов на родительской странице, поэтому начните со смещения размер страницы / 2. Затем добавьте разницу между номером текущей страницы и номером родительской страницы минус один, поскольку первая страница после родительской страницы имеет родительский узел в индексе (размер страницы / 2).

Для узлов 2 и 3 родительский элемент различается в зависимости от режима. В режиме экономии места родителями являются просто узлы 0 и 1 соответственно, поэтому нужно вычесть только с 2. С другой стороны, в режиме строгого двоичного дерева эти узлы являются корнями страницы. из 0 и 1, поэтому их общий родительский элемент вычисляется так же, как описано выше.

Для всех остальных узлов их родительский элемент будет на той же странице, и достаточно разделить их смещение внутри их страницы на 2, не изменяя номер страницы.

Смотрите также

Рекомендации

  1. ^ Камп, Поул-Хеннинг (26.07.2020). "Ты делаешь это неправильно". Очередь ACM.
  2. ^ Наор, Далит; Martel, Charles U .; Матлофф, Норман С. (1991). «Производительность структур приоритетной очереди в среде виртуальной памяти». Comput. Дж. 34 (5): 428–437. Дои:10.1093 / comjnl / 34.5.428.
  3. ^ ван Эмде Боас, П.; Kaas, R .; Зийлстра, Э. (1976). «Разработка и внедрение эффективной очереди приоритетов». Математическая теория систем. 10: 99–127. Дои:10.1007 / BF01683268.
  4. ^ phk.freebsd.dk http://phk.freebsd.dk/B-Heap/. Получено 2019-06-08. Отсутствует или пусто | название = (помощь)

внешняя ссылка