- Классификация двухсторонних очередей
- Основные операции с деками
- Иллюстрация Деке
- Реализация Deque
- Приложения
- Заключение
Углубленный учебник по Deque или двусторонней очереди на C++. Учебник объясняет, что такое Deque, основные операции, реализацию и применение на C++ и Java:
Очередь с двойным окончанием или просто "Deque" - это обобщенная версия очереди.
Разница между Queue и Deque в том, что они не следуют подходу FIFO (First In, First Out). Вторая особенность Deque в том, что мы можем вставлять и удалять элементы как с переднего, так и с заднего конца.
Классификация двухсторонних очередей
Deque можно классифицировать следующим образом:
Deque с ограниченным входом: При ограничении по входу удаление может быть выполнено с обоих концов, но вставка может быть выполнена только в задней части очереди.
Deque с ограничением выхода: В очереди с ограничением выхода вставка может осуществляться с обоих концов, но удаление производится только с одного конца, т.е. с переднего конца очереди.
Мы также можем реализовать стеки и очереди с помощью deque.
Основные операции с деками
Ниже перечислены основные операции, которые можно выполнить над deque.
- вставной фронт: Вставить или добавить элемент в начало deque.
- insertLast: Вставить или добавить элемент в заднюю часть deque.
- deleteFront: Удалить или убрать элемент из первой части очереди.
- удалить последнюю: Удалить или убрать элемент из задней части очереди.
- getFront: Извлекает первый элемент в deque.
- getLast: Извлекает последний элемент в очереди.
- isEmpty: Проверяет, пуст ли deque.
- isFull: Проверяет, заполнен ли deque.
Иллюстрация Деке
Пустой deque представляется следующим образом:
Далее мы вставляем элемент 1 спереди.
Теперь вставляем элемент 3 сзади.
Далее мы добавляем элемент 5 к фронту, и при увеличении фронт указывает на 4.
Затем мы вставляем элементы 7 сзади и 9 спереди. Deque будет выглядеть так, как показано ниже.
Далее давайте удалим элемент с фронта.
Таким образом, мы видим, что когда элементы вставляются спереди, позиция спереди уменьшается, а при удалении элемента она увеличивается. Для задней части позиция увеличивается при вставке и уменьшается при удалении. .
Реализация Deque
Реализация Deque в C++
Мы можем реализовать deque в C++, используя массивы, а также связный список. Кроме того, в стандартной библиотеке шаблонов (STL) есть класс "deque", который реализует все функции для этой структуры данных.
Ниже приведена реализация массива deque. Поскольку это двусторонняя очередь, мы использовали для реализации круговые массивы.
#includeusing namespace std; #define MAX_size 10 // Максимальный размер массива или Dequeue // Класс Deque class class Deque { int array[MAX_size]; int front; int rear; int size; public : Deque(int size) { front = -1; rear = 0; this->size = size; } // Операции над Deque: void insertfront(int key); void insertrear(int key); void deletefront(); void deleteterear(); int getFront(); int getRear(); // Проверка, является ли Dequefull bool isFull() return ((front == 0 && rear == size-1) // Проверяем, пуст ли Deque bool isEmpty(){ return (front == -1); } }; // Вставляем элемент в начало deque void Deque::insertfront(int key) { if (isFull()) { cout <<"Overflow!!\n" <<endl; return; } // Если очередь изначально пуста, устанавливаем front=rear=0; начало deque if (front == -1) { front = 0; rear = 0; } else if(front == 0) // front является первой позицией очереди front = size - 1 ; else // уменьшаем front на 1 позицию front = front-1; array[front] = key ; // вставляем текущий элемент в Deque } // вставляем элемент в задний конец deque void Deque ::insertrear(int key) { if (isFull()) { cout <<" Overflow!!\n " <<endl; return; } // Если очередь изначально пуста, устанавливаем front=rear=0; начало deque if(front == -1) { front = 0; rear = 0; } else if (rear == size-1) // rear находится на последней позиции очереди rear = 0; else // увеличиваем rear на 1 позицию rear = rear+1; array[rear] = key ; // вставляем текущий элемент в Deque } // Удаляем элемент спереди Deque void Deque ::deletefront() { if (isEmpty()) { cout <<"Queue Underflow!!\n" <<endl; return ; } // Deque имеет только один элемент if(front == rear) { front = -1; rear = -1; } else // возвращаемся в исходное положение if (front == size -1) front = 0; else // удаляем текущее значение front из Deque;увеличиваем front на 1 front = front+1; } // Удаляем элемент в заднем конце Deque void Deque::deleterear() { if (isEmpty()) { cout <<" Underflow!!\n" <<endl ; return ; } // Deque имеет только один элемент if (front == rear) { front = -1;rear = -1; } else if (rear == 0) rear = size-1; else rear = rear-1; } // получение переднего элемента Deque int Deque::getFront() { if (isEmpty()) { cout <<" Underflow!!\n" <<endl; return -1 ; } return array[front]; } // получение заднего элемента Deque int Deque::getRear() { if(isEmpty())//главная программа int main() { Deque dq(5); cout <<"Вставить элемент 1 в задний конец \n"; dq.insertrear(1); cout <<"вставить элемент 3 в задний конец \n"; dq.insertrear(3); cout <<"задний элемент deque " <<" <<dq.getRear() <<endl; dq.deleterear(); cout <<"После deleterear, rear = " <<dq.getRear() <<endl; cout <<"вставить элемент 5 вfront end \n"; dq.insertfront(5); cout <<"front element of deque " <<" <<dq.getFront() <<endl; dq.deletefront(); cout <<"After deletefront, front = " <<dq.getFront() <<endl; return 0; }
Выход:
Вставьте элемент 1 в заднюю часть
вставить элемент 3 на заднем конце
задний элемент deque 3
После удаления заднего колеса, заднее =
вставка элемента 5 на переднем конце
передний элемент deque 5
После удаления фронта, фронт =
Реализация Java Deque
Интерфейс deque в Java, "java.util.Deque", является производным от интерфейса "java.util.Queue". Deque можно использовать как очередь (First In, First Out) или стек (Last In, First Out). Эти реализации работают быстрее, чем связный список.
Ниже приведена иерархия для интерфейса Deque в Java.
Мы должны помнить несколько моментов об интерфейсе Deque в Java:
- Реализация не является потокобезопасной, так как отсутствует внешняя синхронизация.
- Deque не поддерживает параллелизм нескольких потоков.
- Deque, реализованные с использованием массивов, не позволяют использовать NULL-элементы.
- Массивы могут расти в соответствии с требованиями, при этом двумя наиболее важными характеристиками являются свободная от ограничений емкость и поддержка изменяемых размеров массива.
Ниже перечислены различные методы, поддерживаемые интерфейсом Deque:
Нет. | Метод | Описание |
---|---|---|
1 | add(element) | Добавляет элемент в хвост. |
2 | addFirst(element) | Добавляет элемент к голове/переду. |
3 | addLast(element) | Добавляет элемент в хвост/заднюю часть. |
4 | предложение(элемент) | Добавляет элемент в хвост; возвращает булево значение, указывающее на успешность вставки. |
5 | offerFirst(element) | Добавляет элемент в голову; возвращает булево значение, указывающее на успешность вставки. |
6 | offerLast(element) | Добавляет элемент в хвост; возвращает булево значение, указывающее на успешность вставки. |
7 | итератор() | Возвращает итератор для deque. |
8 | descendingIterator() | Возвращает итератор, имеющий обратный порядок для данного deque. |
9 | push(element) | Добавляет элемент в начало deque. |
10 | pop(element) | Удаляет элемент из головы deque и возвращает его. |
11 | removeFirst() | Удаляет элемент, находящийся в начале deque. |
12 | removeLast() | Удаляет элемент, находящийся в хвосте deque. |
13 | poll() | Извлекает и удаляет первый элемент deque (представленный головой deque); возвращает NULL, если deque пуст. |
14 | pollFirst() | Извлекает и удаляет первый элемент данного deque; возвращает null, если deque пуст. |
15 | pollLast() | Извлекает и удаляет последний элемент данного deque; возвращает null, если deque пуст. |
16 | peek() | Извлекает голову (первый элемент deque) очереди, представленной этой deque; возвращает null, если эта deque пуста. Примечание: Эта операция не удаляет элемент. |
17 | peekFirst() | Извлекает первый элемент данного deque; возвращает null, если deque пуст. Примечание: Эта операция не удаляет элемент. |
18 | peekLast() | Извлекает последний элемент этого deque или возвращает null, если этот deque пуст. Примечание: Эта операция не удаляет элемент. |
Следующая реализация на Java демонстрирует различные операции, рассмотренные выше.
import java.util.*; class Main { public static void main(String[] args) { Dequedeque = новый LinkedList (); // Мы можем добавлять элементы в очередь различными способами deque.add(1); // добавить в хвост deque.addFirst(3); deque.addLast(5); deque.push(7); // добавить в голову deque.offer(9); deque.offerFirst(11); deque.offerLast(13); System.out.println("The deque : " + deque + "\n"); // Итерация элементов очереди. System.out.println("Стандартный итератор"); Iterator iterator = deque.iterator(); while(iterator.hasNext()) System.out.print(" " + iterator.next()); // Итератор обратного порядка Iterator reverse = deque.descendingIterator(); System.out.println("\nReverse Iterator"); while (reverse.hasNext()) System.out.print(" " + reverse.next()); // Peek возвращает голову, не удаляя // ее из deque System.out.println("\n\nPeek " + deque.peek()); System.out.println("After peek: " + deque);// Pop возвращает голову и удаляет ее из deque System.out.println("\nPop " + deque.pop()); System.out.println("After pop: " + deque); // Мы можем проверить, существует ли определенный элемент // в deque System.out.println("\nContains element 3?: " + deque.contains(3)); // Мы можем удалить первый / последний элемент. deque.removeFirst(); deque.removeLast(); System.out.println("Deque после удаления "+ "первый и последний элементы: " + deque); } }
Выход:
Дека : [11, 7, 3, 1, 5, 9, 13].
Стандартный итератор
11 7 3 1 5 9 13
Обратный итератор
13 9 5 1 3 7 1
Взгляд 11
После пика: [11, 7, 3, 1, 5, 9, 13].
Поп 11
После хлопка: [7, 3, 1, 5, 9, 13].
Содержит элемент 3?: true
Deque после удаления первого и последнего элементов: [3, 1, 5, 9]
В приведенной выше программе мы использовали интерфейс Deque языка Java и определили deque из целочисленных элементов. Затем мы выполнили различные операции над этим deque и вывели результаты этих операций на экран.
Приложения
Deque может использоваться в некоторых из следующих приложений.
#1) Алгоритм составления расписания: Алгоритм планирования "A-steal scheduling algorithm" реализует планирование задач для различных процессоров в многопроцессорной системе. Эта реализация использует deque, и процессор получает первый элемент из deque для выполнения.
#2) Отменить список действий: В программных приложениях у нас есть множество действий. Одно из них - "отмена". Когда мы выполняем действие отмены много раз, все эти действия сохраняются в списке. Этот список ведется как deque, чтобы мы могли легко добавлять/удалять записи с любого конца.
#3) Удалите записи через некоторое время: Приложения обновляют записи в своем списке, например, приложения, отображающие записи о запасах и т.д. Эти приложения удаляют записи через некоторое время, а также вставляют новые записи. Это делается с помощью deque.
Заключение
Deque - это двухсторонняя очередь, которая позволяет нам добавлять/удалять элементы с обоих концов очереди, т.е. спереди и сзади. Deque может быть реализован с помощью массивов или связанных списков. Однако у нас также есть класс стандартной библиотеки шаблонов (STL), который реализует различные операции Deque.
В Java есть интерфейс Deque, который наследуется от интерфейса очереди для реализации Deque. Помимо основных стандартных операций Deque, этот интерфейс поддерживает различные другие операции, которые могут быть выполнены над Deque.
Deque обычно используется в приложениях, требующих добавления/удаления элементов с обоих концов. Он также в основном используется при планировании работы процессоров в многопроцессорных системах.
Ознакомьтесь с полным учебным циклом по C++