List order maintenance
List order maintance (рус. поддержка порядка в списке) — проблема поддержки списка со следующими операциями:
- — вставка нового элемента в список сразу после ;
- — удаление элемента из списка;
- — команда, возвращающая , если в списке находится до и иначе.
Существует реализация такой структуры, где персистентного дерева поиска.
выполняется за истинную, а команды добавления и удаления за амортизационную . Проблема поддержки порядка в списке возникает, к примеру, при реализацииАлгоритм
Идея
Все операции кроме двусвязный список, но с его помощью невозможно получить информацию о порядке объектов. Чтобы реализовать эту операцию, каждому узлу будет сопоставлено некоторое число так, чтобы все числа строго возрастали от начала к концу списка. Таким образом, эти числа, которые в дальнейшим будут называться метками, задают порядок на элементах списка.
за может выполнить обычныйОтветить на запрос
можно за , просто сравнив метки и . Добавление меток никак не влияет на реализацию операции . Однако реализацию потребуется изменить: при добавлении нового элемента после узла , узлу необходимо присвоить метку, которая строго больше предыдущего элемента и строго меньше следующего. Исходя из предположения, что метки имеют конечную длину, в какой-то момент возникнет ситуация, что новой метки не найдётся, тогда метки будут перераспределены среди элементов списка так, что хватит метки. Далее будет рассмотрен алгоритм, который позволяет эффективно реализовать эту идею.Алгоритм за O(logn)
Способ хранения меток
Метки будут храниться в виде чисел в двоичной системе счисления. Требуется выбрать такую длину для меток, чтобы перераспределения не случались слишком часто. Если
— длина каждой метки, то для начала пусть , где — количество элементов в списке. Если после добавления или удаления элементов перестанет удовлетворять неравенству, пересчитаем все метки заново. Пересчет меток занимает амортизационно по аналогии с саморасширяющимся массивом. Позже, в доказательстве времени работы, значение будет несколько уточнено.Все метки будут храниться в цифровом боре высоты (там представлены не только используемые метки, а вообще все возможные заданной длины). Также в каждом узле дополнительно будет храниться:
- в листьях — используется ли уже эта метка;
- в нелистовых узлах — является ли узел переполненным.
Переполненным является узел, для которого для любой
выполнено , где — это количество помеченных (используемых) листьев (меток) в поддереве узла , а — это количество всех листьев в поддереве . В листьях мы не храним наличие переполненности, так как все листья всегда непереполнены. В крайнем случае: .Перераспределение меток
Перераспределение меток потребуется выполнить в случае, когда для нового элемента нет свободной метки. Ниже будет описан способ выполнения перераспределения меток: как только требуется вставить элемент
, которому не хватает метки, будем подниматься вверх от метки элемента, следующего за , пока не будет найден первый непереполненный узел. Может случиться такое, что на всем пути до корня не встретится ни одного непереполненного узла. Чтобы этого избежать, границы будут уточнены позже. Как только будет найден первый непереполненный узел, переназначим метки в его поддереве так, чтобы они находились друг от друга на одинаковых расстояниях (места точно хватит, так как , если узел непереполненный). После этого между занятыми метками будет не меньше свободных меток.Доказательство времени работы
Ниже будут рассмотрены операции добавления более подробно, чтобы оценить, как часто происходит перераспределение меток:
- Если в поддереве узла было проведено перераспределение меток, то повторное перераспределение меток в поддереве узла потребуется, когда сын этого узла снова переполнится. Если — сын , то он переполнится, когда . Чтобы это произошло, требуется, чтобы было сделано еще добавлений;
- С другой стороны, следующее перераспределение меток произойдет, когда .
Получается, что за
операций перераспределения меток требуется сделать операций добавления. Из этого следует, что, если за каждую операцию добавления брать монет, то за добавления накопится достаточное количество монет, чтобы расплатиться за следующую операцию перераспределения в узле .Однако таким образом требуется платить за каждый уровень, а количество уровней (бит) равно
. Тогда амортизированная стоимость добавления .Далее будут уточнены границы
, чтобы корень никогда не переполнялся: . Тогда операция добавления работает за .Алгоритм за O(1)
Предыдущий алгоритм работает за логарифм из-за того, что слишком часто приходится делать перераспределение меток. Используем
(модифицированный цифровой бор), чтобы улучшить время работы операции добавления до .У каждого элемента списка будет две метки: глобальная и локальная. Глобальная задает блок, локальная — положение элемента внутри блока. Описание взаимодействия с метками:
- локальные метки внутри каждого блока будут присвоены каждому элементу от до жадно. Стоит заметить, что внутри блока никогда не будет проблемы, что кому-то не хватит метки или придется сделать перераспределение меток, так как, если каждый раз в качестве метки брать среднее значение, то для того, чтобы был конфликт из-за меток, нужно больше, чем ключей (противоречит условию);
- глобальные метки будут организованы в структуру, использовавшуюся в реализации операции за логарифмическое время. Глобальные метки для блоков придется менять, когда один из блоков переполнился. Тогда блок будет разделен на два, метка второму будет присвоена методом, описанным выше (поднимемся до первого непереполненного). Каждый блок будет иметь занятых меток. Аналогично, когда в каком-то блок становится слишком мало ключей, он будет слит с соседним.
Внутри блоков присваиваются ключи за
, а, аналогичный приведенному выше анализ показывает, что, чтобы потребовалось перераспределение глобальных меток, требуется изменений локальных меток. За эти изменения будет накоплено монет для изменения глобальных меток, тогда операция добавления работает за константное время.Использование памяти
Из-за того, что
зависит от выбранной , сильно влияет на реализацию. Увеличивая , уменьшается стоимость операции добавления (количество монет, которые надо брать: ), но увеличивается , значит, требуется больше памяти, а, уменьшая , возникает выигрыш в памяти, но проигрыш во времени операции добавления. Так как для реализации структуры используется , требуется памяти.Послесловие
Впервые реализацию такой структуры данных со всеми операциями за константное время амортизационно предложили[1] Dietz и Sleator, однако их доказательство времени работы было намного сложнее вышеизложенного анализа. Поэтому позже группа ученых во главе с Michael A. Bender разработала[2] более простое доказательство, изложенное выше, впервые описанное в их статье Two simlified algorithms for maintaining order in a list. Послесловие их статьи таково:
Dietz and Sleator is quite influential With its tags and its proofs by potential But to teach it in class Is a pain in the ass So our new result is preferential.