C++のDeque(ダブルエンド型キュー)についての詳細なチュートリアルです。 Dequeとは何か、基本操作、C++とJavaの実装と応用について説明しています:
ダブルエンドキュー(Deque)とは、キュー(Queue)を一般化したもので、「Deque」と呼ばれる。
QueueとDequeの違いは、FIFO(First In, First Out)方式ではないことです。 Dequeの2つ目の特徴は、前端と後端のどちらからでも要素の挿入・削除が可能なことです。
ダブルエンド型待ち行列の分類
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の実装
C++ Dequeの実装
C++では、リンクリストと同様に配列を使ってdequeを実装することができます。 これとは別に、標準テンプレートライブラリ(STL)には、このデータ構造に関するすべての機能を実装した「deque」クラスがあります。
以下に、Dequeの配列実装を示します。 ダブルエンドのキューであるため、実装には円形配列を使用しています。
#インクルードusing namespace std; #define MAX_size 10 // 配列またはDequeueの最大サイズ // Dequeクラス 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 deleterear(); int getFront(); int getRear(); // Dequeかどうかをチェック。full bool isFull() return ((front == 0 && rear == size-1) // Dequeが空かどうかチェック bool isEmpty(){ return (front == -1); } }; // Dequeの先頭に要素を挿入 void Deque::insertfront(int key) { if (isFull()) { cout <<"Overflow!"<<endl; return; } // キューが最初は空の場合, set 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 <<" オーバーフロー!!⑭ " <<endl; return; } // キューが最初は空なら set 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!"<<endl; return ; } // Dequeには要素が一つしかありません if(front == rear) { front = -1; rear = -1; } else // 初期位置に戻る if (front == size -1) front = 0; else // Dequeから現在のフロント値を削除し、フロントを1つ増分する front = front+1; } // Dequeの後端の要素を削除 void Deque::deleterear() { if (isEmpty()) { cout <<" アンダーフロー!!ꉂ(ˊᗜˋ*) <endl ; } // Doqueは要素が1つだけ 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 <<" アンダーフロー!!୧⃛(๑⃙⃘◡̈︎๑⃙⃘) <endl; return -1 ; } return array[front]; } // Dequeの後ろ要素を取得 int Deque::getRear() { if(isEmpty()//main program int main() { Deque dq(5); cout <<"後端に要素1を挿入"; dq.insertrear(1); cout <<"後端に要素3を挿入"; dq.insertrear(3); cout <<"Deque の後部要素 " <<" <dq.getRear() <Endl; cout <<"deleterear 後部 = " <<dq.getRear() <Endl; cout <<"挿入要素 5 atdq.insertfront(5); cout <<"dequeの前要素 " <<" <<dq.getFront() <<endl; dq.deletefront(); cout <<"After deletefront, front = " <<dq.getFront() <<endl; return 0; }"
出力します:
後端部にエレメント1を挿入する
後端部に要素3を挿入する
deque 3 の後段要素
deleterear後、リア=。
フロントエンドに要素5を挿入する
表5
deletefrontの後、front =
Java Dequeの実装
Javaのdequeインターフェイス「java.util.Deque」は、「java.util.Queue」インターフェイスから派生しました。 Dequeは、キュー(先入れ先出し)またはスタック(後入れ先出し)として使用できます。 これらの実装はリンクリストよりも高速に動作します。
JavaのDequeインターフェイスの階層を以下に示します。
JavaのDequeインターフェイスについて、いくつかのポイントを覚えておく必要があります:
- 外部同期を行わないため、スレッドセーフな実装ではありません。
- Dequeは複数スレッドによる並行処理をサポートしていません。
- 配列を使って実装されたDequeでは、NULLの要素を使用することはできません。
- アレイは要件に応じて成長させることができ、制限のない容量とリサイズ可能なアレイサポートが2大特徴です。
以下は、Deque インターフェースがサポートする各種メソッドです:
いいえ。 | 方法 | 商品説明 |
---|---|---|
1 | 追加要素 | 尾部に要素を追加する。 |
2 | かそくどてき | ヘッド/フロントに要素を追加する。 |
3 | addLast(要素) | 尾部/後部に要素を追加する。 |
4 | 供す | 要素を末尾に追加し、挿入が成功したかどうかを示すブーリアン値を返す。 |
5 | はんにゅう | 要素をheadに追加し、挿入が成功したかどうかを示すboolean値を返す。 |
6 | offerLast(要素) | 要素を末尾に追加し、挿入が成功したかどうかを示すブーリアン値を返す。 |
7 | 反復子() | dequeのイテレータを返します。 |
8 | descendingIterator() | このDequeの逆順位を持つイテレータを返す。 |
9 | 押し出す | dequeの先頭に要素を追加する。 |
10 | ポップ | dequeの先頭から1つの要素を削除して返す。 |
11 | removeFirst() | dequeの先頭にある要素を削除します。 |
12 | removeLast() | dequeの末尾にある要素を削除します。 |
13 | ポーリング() | deque の最初の要素(deque の head で表される)を取得し、削除する。deque が空の場合は NULL を返す。 |
14 | pollFirst() | このDequeの最初の要素を取得し、削除する。このDequeが空の場合はNULLを返す。 |
15 | pollLast() | このDequeの最後の要素を取得し、削除する。このDequeが空の場合はNULLを返す。 |
16 | 覗き見 | このDequeで表されるキューのhead(Dequeの最初の要素)を取得します。 注:この操作では、エレメントは取り外せません。 |
17 | peekFirst()する | このDequeの最初の要素を取得する。このDequeが空である場合はNULLを返す。 注:この操作では、エレメントは取り外せません。 |
18 | peekLast() | このDequeの最後の要素を取得し、このDequeが空の場合はNULLを返します。 注:この操作では、エレメントは取り外せません。 |
以下のJavaの実装は、上記で説明した様々な操作のデモンストレーションを行います。
import java.util.*; class Main { public static void main(String[] args) { Dequedeque = new 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("Standard 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("\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("After removing")+ "最初と最後の要素: " + deque); } } } 。
出力します:
デク : [11, 7, 3, 1, 5, 9, 13] です。
標準イテレータ
11 7 3 1 5 9 13
リバースイテレーター
13 9 5 1 3 7 1
Peek 11
ピークの後:【11、7、3、1、5、9、13】。
ポップ11
ポップ後:[7、3、1、5、9、13]。
要素3を含むか:true
最初と最後の要素を削除した後のDeque: [3, 1, 5, 9] 。
上記のプログラムでは、JavaのDequeインターフェイスを利用して、整数要素のDequeを定義し、このDequeに対して様々な演算を行い、その結果を出力しています。
アプリケーション
Dequeは、以下のような用途で使用することができます。
#その1)スケジューリング・アルゴリズム: スケジューリングアルゴリズム "A-steal scheduling algorithm "は、マルチプロセッサシステムにおける様々なプロセッサのタスクスケジューリングを実装する。 この実装では、dequeを使用し、プロセッサはdequeから最初の要素を取得し実行する。
#2)アクティビティ一覧を元に戻す: ソフトウェアアプリケーションでは、様々なアクションがありますが、その中の一つに「アンドゥ」があります。 アンドゥを何度も行った場合、これらのアクションは全てリストに格納されます。 このリストはdequeとして管理され、どこからでも簡単にエントリーを追加/削除できるようになっています。
#3) しばらくしてから、エントリーを削除してください: これらのアプリは、しばらくするとエントリーを削除し、新しいエントリーを挿入します。 これは、dequeを使用して行われます。
結論
Dequeはダブルエンド型のキューで、キューの前後両端から要素を追加・削除することができます。 Dequeは配列やリンクリストを使って実装できますが、Dequeのさまざまな操作を実装する標準テンプレートライブラリ(STL)クラスも用意されています。
Javaでは、Dequeを実装するために、queueインターフェイスを継承したDequeインターフェイスを用意しています。 Dequeの基本的な標準操作以外に、Dequeに対して実行可能な様々な操作をこのインターフェイスはサポートしています。
Dequeは一般に、両端から要素を追加/削除する必要があるアプリケーションに使用されます。 また、マルチプロセッサシステムにおけるプロセッサのスケジューリングに多く使用されます。
C++トレーニングシリーズをチェックする