Java ソートアルゴリズムの概要 (VII): クイックソート

Java ソートアルゴリズムの概要 (VII): クイックソート

クイックソートはバブルソートの改良版です。その基本的な考え方は、ソート パスを通じて、ソートするデータが 2 つの独立した部分に分割され、一方の部分のすべてのデータがもう一方の部分のすべてのデータよりも小さくなり、その後、この方法に従って 2 つの部分のデータが別々にすばやくソートされるというものです。ソート プロセス全体を再帰的に実行して、データ全体を順序付けられたシーケンスに変換できます。クイックソートは不安定で、O(log(n)) の追加スペースを必要とし、時間計算量は O(nlog(n)) であり、適応性がありません。

クイックソートには、言及する価値のあるバリエーションがいくつかあります。以下に簡単に紹介します。

ランダム化されたクイックソート:クイックソートの最悪のケースは、各パーティションのピボットの選択に基づきます。基本的なクイックソートでは、最初の要素がピボットとして選択されます。このように、配列がすでに順序付けられている場合、各分割は最悪の結果になります。一般的な最適化方法は、要素をメイン要素としてランダムに選択するランダム化アルゴリズムです。この場合、最悪のケースは依然として O(n^2) ですが、最悪のケースはもはや入力データに依存せず、ランダム関数の評価が不十分なために発生します。実際、クイックソートをランダム化して理論上の最悪のケースを得る確率は 1/(2^n) だけです。したがって、ランダム化クイックソートは、ほとんどの入力データに対して予想される時間計算量 O(nlogn) を達成できます。ある先輩が素晴らしい要約をしてくれました。「ランダムクイックソートは、人の性格のニーズを一生涯満たすことができます。」

ランダム化クイックソートの唯一の欠点は、入力データに同一データが大量に存在すると、ランダム化の効果が直接的に弱まってしまうことです。極端なケース、つまり n 個の同一の数字をソートする場合、ランダムクイックソートの時間計算量は間違いなく O(n^2) に削減されます。解決策は、ピボットを交換せずにそのままの状態でスキャンすることです。

バランス クイックソート: 毎回、中央値をキー データとして表すことができる要素を選択し、通常のクイックソートの原則に従って比較、置換、再帰を実行します。一般的に、このデータを選択する方法は、開始データ、終了データ、中間データを取得し、比較して中央値を選択することです。この3つの値を取る利点は、実際の問題(情報科学のコンテストなど)では、近似連続データや反転データが出現する可能性が高いことです。この場合、真ん中のデータが必然的に中央値になり、実は近似中央値でもあります。中央が大きく両側が小さい(またはその逆)データに遭遇し、取得した値が最大値に近い場合、少なくとも2つの部分を分離できるため、実際の効率は約2倍になり、データを少し乱して縮退構造を破壊することが有益です。

外部クイックソート: 通常のクイックソートとは異なり、キーデータはバッファです。まず、前と次の M/2 要素がバッファに読み込まれ、バッファ内のこれらの要素がソートされます。次に、ソートされた配列の先頭 (または末尾) から次の要素が読み込まれます。この要素がバッファ内の最小の要素よりも小さい場合は、最初の空きスペースに書き込まれます。この要素がバッファ内の最大の要素よりも大きい場合は、最後の空きスペースに書き込まれます。それ以外の場合は、バッファ内の最大または最小の要素が配列に書き込まれ、バッファに配置されます。すでに順序付けされた中間データの再配置を避けるため、最大値はこれらのキー データより低く、最小値はこれらのキー データより高く設定してください。完了後、配列の中央のスペースを空ける必要があり、バッファは配列の中央のスペースに書き込まれます。次に、小さい外側の部分を再帰的にソートし、他の部分を循環的にソートします。

3 方向基数クイックソート (マルチキー クイックソート、マルチキー クイックソートとも呼ばれます) : 基数ソート (一般的な文字列比較ソートは基数ソートであるなど) とクイックソートの特性を組み合わせたもので、文字列ソートに比較的効率的なアルゴリズムです。このアルゴリズムによってソートされる配列の要素には、文字列などのマルチキーという特性があり、各文字はキーとみなすことができます。アルゴリズムは、ソートされた配列内の要素をキー データとしてランダムに選択するたびに、まずこの要素の最初のキー (文字) のみを考慮し、次にキーを比較して、他の要素をキー データより小さい、等しい、大きい 3 つの部分に分割します。次に、このキーの位置に基づいて「より小さい」部分と「より大きい」部分を再帰的にソートし、次のキーに基づいて「等しい」部分をソートします。

コード実装:

  1. 公共 クラスクイックソート{
  2. 公共 静的  void sort(比較可能なデータ[]、 int low、 int high) {
  3. // ピボット要素。通常は最初の要素に基づいて分割されます 
  4. int i = 低い;
  5. 整数j = 高い;
  6. (低<高)の場合{
  7. // 配列の両端から中央まで交互にスキャンします 
  8. 比較可能な pivotKey = data[low];
  9. // ポインタ i、j をスキャンします。i は左から始まり、j は右から始まります 
  10. i < j である間{
  11. i < j && data[j].compareTo(pivotKey) > 0場合
  12. j--;
  13. } // 終了 
  14. もし(i < j){
  15. // ピボット要素より小さい要素を左に移動する 
  16. データ[i] = データ[j];
  17. 私は++;
  18. } // 終了 
  19. i < j && data[i].compareTo(pivotKey) < 0場合
  20. 私は++;
  21. } // 終了 
  22. もし(i < j){
  23. // ピボット要素より大きい要素を右に移動する 
  24. データ[j] = データ[i];
  25. j--;
  26. } // 終了 
  27. } // 終了 
  28. // ピボット要素を正しい位置に移動する 
  29. データ[i] = pivotKey;
  30. // サブリストの前半を再帰的にソートする 
  31. ソート(データ、low、i - 1 );
  32. // サブリストの後半部分を再帰的にソートする 
  33. ソート(データ、i + 1 、高);
  34. } // 終了 
  35. } // ソート終了 
  36. 公共 静的  void main(String[] args) {
  37. // JDK1.5以降では、基本データ型は自動的にボックス化されます 
  38. // intやdoubleなどの基本型のラッパークラスはComparableインターフェースを実装しています 
  39. 比較可能[] c = { 4 9 23 1 45 27 5 2 };
  40. ソート(c, 0 , c.length - 1 );
  41. (比較可能データ: c) {
  42. System.out.println(データ);
  43. }
  44. }
  45. }

クイックソートは現在最も広く使われているソートアルゴリズムです。クイックソートの核心はセグメンテーションアルゴリズムにあり、最も巧妙な部分とも言えます。この記事がお役に立てば幸いです。

【編集者のおすすめ】

  1. 5.11 3行のコードでクイックソート
  2. 12.3.2 タスクスティールに基づくクイックソート
  3. 19.4.2 Parallel_For による並列クイックソート
  4. C# クイックソートの興味深い実装: Haskell から始める

<<:  Java ソートアルゴリズムの概要 (VIII): 基数ソート

>>:  Java ソートアルゴリズムの概要 (VI): ヒープソート

ブログ    
ブログ    
ブログ    
ブログ    
ブログ    

推薦する

グレートウルフホテルはAIを活用してゲストの体験とレビューを理解する

現在、ホテルやエンターテインメント業界のチェーンは、ゲストの体験やレビューをスキャンして理解するため...

指紋と顔の認識が手のひらスキャンにアップグレードされ、大ヒット映画でしか見られない新技術がシティエキスポでデビュー

[[250312]]手のひらをスワイプするだけで入場や支払いができ、道路清掃車にセンサーを追加するこ...

JVMシリーズ(3):GCアルゴリズムガベージコレクター

[[204469]]概要ガベージコレクションは、通常「GC」と呼ばれます。1960年にMITのLis...

1780億のパラメータを持つこの言語モデルは、王者GPT-3に挑戦するためだけに作られたのでしょうか?

誰かがGPT-3の独占に挑戦しなければなりません! GPT-3 は発売以来、最大の AI 言語モデル...

ロボットが人間のように学習できるようにする Google RT-2 AI モデルとは何ですか?

ビッグモデルが急増し、仮想世界から現実世界に進出しています。 Google DeepMind は最近...

コードコーパス、大規模モデル、インテリジェントエージェントの魔法の杖を振ると、より強力なエネルギーが呼び出されます

熱帯雨林の杖が、ダンブルドアのようなあらゆる時代の並外れた魔法使いの伝説を生み出したのと同じように、...

...

9月30日付けでマイクロソフトがAIサービス規約を更新:リバースエンジニアリング等に利用不可

マイクロソフトは8月16日、AI利用規約を発表し、9月30日に正式に発効すると発表した。新しい用語は...

ChatGPTを超える最初のオープンソースモデルが登場?ネットユーザーはそれを信じない

大型モデルが人気となり、毎日さまざまな「ビッグ」ニュースを目にするようになりました。写真今日、もう一...

ドローン時代の到来により、人工知能航空機が有人戦闘機に取って代わり、パイロットは失業することになるのでしょうか?

まず、ドローンはソレイマニの暗殺に使用され、その後、アルメニアとアゼルバイジャンの戦場でドローンが活...

メーター読み取りシステムにおける無線データ伝送モジュールの応用

周知のとおり、従来の手動メーター読み取り方法は時間がかかり、労働集約的であり、その正確性と適時性は保...

このクラウドは、AIが後半にどのように発展するかを知っている

今年はAI技術の導入が話題になっています。 AIは本当に実装されているのでしょうか?真実を語るには実...

...

...

ニューラルネットワークはとてもシンプルです。機械学習の入門書をご紹介します | 役立つ情報

[[331060]]この記事はAI新メディアQuantum Bit(公開アカウントID:QbitAI...