Персистентный дек
Дек (англ. deque — double ended queue — очередь с двумя концами) — структура данных с двусторонним доступом к элементам, т.е. их можно удалять и добавлять как в начало, так и в конец дека.
Кроме дека ещё существует структура данных, называемая steque, которая представляет собой объединение стека и очереди - элементы можно добавлять только в один конец, а извлекать — с обоих.
Эффективная реализация
Персистентный дек можно визуально представить как дерево, где каждый узел хранит пару - левый элемент и правый, а также ребёнка - ссылку на следующий дек. Только с каждым уровнем вложенности левый и правый элемент хранят в два раза больше объектов, чем на предыдущем уровне.
Тип
Pair<> { };
Сам дек можно инициализировать напрямую, вызвав конструктор
Deque<> { left; right; Deque<Pair< >> child; };
Таким образом элементы с начала хранятся в левой ветке дерева, а с конца - в правой.
Наша структура данных персистентна, следовательно операция
push_front(x, D) if D ==// если дек пустой, то формируем новый дек return Deque(x, ) else if D.left == // если левый ребенок не существует, то сделаем левым ребёнком return Deque(x, D.child, D.right) else // иначе объединим левого ребёнка с новым элементом и попытаемся добавить в дек на следующем уровне return Deque( , push_front(Pair(x, D.left), D.child), D.right)
Метод
pop_front(D) if D ==// изъятие элемента из пустого дека возвращает пару "нулевой элемент — пустой дек" return else if D.left // если левый ребёнок не пуст, то возвращаем пару из него и нового дека без левого ребёнка, // но если остался только левый ребёнок, но возвращаем его и пустой дек if D.child = and D.right = return D.left, return D.left, Deque( , D.child, D.right) else if D.child == // если левый ребёнок оказался пуст, и при этом ссылка на следующий дек отсутствует, // то вернём пару из правого ребёнка и абсолютно пустого дека return D.right, else /* * если два предыдущих условия оказались не выполнены, то мы рекурсивно вызываем метод pop_front() * и возвращённую пару "элемент — новый дек" сохраняем в переменные temp & newDeque * Рекурсивные вызовы прекратятся, как только левый ребёнок окажется существующим * или в деке будет отсутствовать ссылка на следующий дек */ temp, newDeque pop_front(D.child) if temp == /* * это возможно только тогда, когда в деке на максимальной глубине все элементы оказались пусты, * значит, мы сейчас на предпоследнем уровне, левый ребёнок пустой и child ссылается на абсолютно пустой дек, * поэтому возвращаем right текущего дека и пустой дек */ return D.right, else /* * если всё же temp не пуст, то надо вернуть первый элемент пары temp; * в качестве left нового дека надо поставить temp.last (на уровне ниже temp хранил * в два раза больше элементов, поэтому на текущем уровне temp.last будет соответствовать * требуемому количеству элементов); newDeque делаем child'ом * нового дека, а right текущего right'ом нового */ return temp.first, Deque(temp.last, newDeque, D.right)
Операции добавления в правый конец и извлечение из него делаются симметрично.
Таким образом, если мы добавляем элементы только в один конец, то на
-ом уровне дека не более элементов. Пусть глубина текущего дека Тогда в нём может находится не более объектов, откуда получаем .Чтобы извлечь элемент, придётся спуститься не больше, чем на глубину дерева. Аналогично для добавления. Поэтому обе операции выполняются за
Пример
Рассмотрим поподробнее операции. В худшем случае элементы будут добавляться только в один конец, а извлекаться из другого.
push_front
Изначально у нас пустой дек. Элементы будем добавлять в левый конец дека и нумеровать согласно порядку добавления. Сначала добавим первый элемент. Он встанет на позицию левого ребёнка первого уровня дека. Теперь попытаемся добавить второй элемент. Позиция левого ребёнка занята, значит, мы объединяем новый элемент со старым и ставим сформированную пару на место левого ребёнка второго дека. Процесс добавления можно представить, как прибавление 1 к разряду числа в двоичном представлении: если в разряде 1, то подряд идущие за ней единицы обнулятся — элементы объединяются в пары, иначе становятся на место этого разряда.
pop_front
Посмотрим, как работает pop_front на примере дека, симметричного нашему предыдущему (в рисунке надо понять, что элементы хранятся слева направо: 4 3 2 1).
Запустим первый
. Он будет рекурсивно вызывать сам себя, пока не дойдёт до последней ветки. Тогда в пару сохранятся две пары элементов и пустой дек высоты 0. Так как не пуст, то в предыдущий рекурсивный вызов вернётся пара: в temp новый сохранится пара 4 3, на текущем уровне будет пара 2 1, а будет ссылаться на пустой дек. Потом пара 4 3 передастся выше, child сохраняет ссылку на предыдущий дек, и, наконец, 4 извлечётся. И так далее. Аналогично добавлению, процесс извлечения элемента можно представить, как вычитание 1 из младшего разряда двоичного числа.