アルゴリズム

[C言語] ポインタを用いてマージソートを実装する方法を解説

マージソートは、分割統治法を用いた効率的なソートアルゴリズムです。

C言語でマージソートを実装する際には、ポインタを活用して配列の要素を操作します。

まず、配列を再帰的に分割し、各部分をソートします。

次に、ポインタを用いてソートされた部分をマージし、元の配列に統合します。

この方法により、安定したソートが可能となり、時間計算量はO(n log n)です。

ポインタを使うことで、メモリ効率を高め、柔軟な配列操作が可能になります。

ポインタを用いたマージソートの実装

マージソートは、分割統治法を用いた効率的なソートアルゴリズムです。

ここでは、ポインタを用いてマージソートを実装する方法を解説します。

マージソートの分割処理

分割処理のアルゴリズム

マージソートの分割処理は、配列を再帰的に半分に分割していくことにより実現されます。

以下にその基本的なアルゴリズムを示します。

  • 配列の中央を計算し、配列を2つの部分に分割する。
  • 各部分を再帰的に分割し、最小単位(要素が1つになるまで)にする。

ポインタを用いた分割の実装

ポインタを用いることで、配列の部分を直接操作し、効率的に分割を行うことができます。

以下にC言語での実装例を示します。

#include <stdio.h>
#include <stdlib.h>
// 配列を分割する関数
void mergeSort(int *array, int left, int right) {
    if (left < right) {
        int middle = left + (right - left) / 2;
        // 左側を再帰的に分割
        mergeSort(array, left, middle);
        // 右側を再帰的に分割
        mergeSort(array, middle + 1, right);
    }
}

マージソートの統合処理

統合処理のアルゴリズム

分割された配列を統合する際には、各部分を比較しながら新しい配列に要素をコピーしていきます。

以下がその基本的なアルゴリズムです。

  • 左右の部分配列の先頭要素を比較し、小さい方を新しい配列に追加する。
  • すべての要素が統合されるまでこの操作を繰り返す。

ポインタを用いた統合の実装

ポインタを用いることで、配列の要素を効率的に統合することができます。

以下にC言語での実装例を示します。

#include <stdio.h>
#include <stdlib.h>
// 配列を統合する関数
void merge(int *array, int left, int middle, int right) {
    int i, j, k;
    int n1 = middle - left + 1;
    int n2 = right - middle;
    // 一時配列を作成
    int *L = (int *)malloc(n1 * sizeof(int));
    int *R = (int *)malloc(n2 * sizeof(int));
    // データを一時配列にコピー
    for (i = 0; i < n1; i++)
        L[i] = array[left + i];
    for (j = 0; j < n2; j++)
        R[j] = array[middle + 1 + j];
    // 一時配列を統合
    i = 0;
    j = 0;
    k = left;
    while (i < n1 && j < n2) {
        if (L[i] <= R[j]) {
            array[k] = L[i];
            i++;
        } else {
            array[k] = R[j];
            j++;
        }
        k++;
    }
    // 残りの要素をコピー
    while (i < n1) {
        array[k] = L[i];
        i++;
        k++;
    }
    while (j < n2) {
        array[k] = R[j];
        j++;
        k++;
    }
    // メモリを解放
    free(L);
    free(R);
}

完全なマージソート関数の実装

分割と統合の処理を組み合わせて、完全なマージソート関数を実装します。

以下にその例を示します。

#include <stdio.h>
#include <stdlib.h>
// 配列を統合する関数
void merge(int *array, int left, int middle, int right) {
    int i, j, k;
    int n1 = middle - left + 1;
    int n2 = right - middle;
    // 一時配列を作成
    int *L = (int *)malloc(n1 * sizeof(int));
    int *R = (int *)malloc(n2 * sizeof(int));
    // データを一時配列にコピー
    for (i = 0; i < n1; i++) L[i] = array[left + i];
    for (j = 0; j < n2; j++) R[j] = array[middle + 1 + j];
    // 一時配列を統合
    i = 0;
    j = 0;
    k = left;
    while (i < n1 && j < n2) {
        if (L[i] <= R[j]) {
            array[k] = L[i];
            i++;
        } else {
            array[k] = R[j];
            j++;
        }
        k++;
    }
    // 残りの要素をコピー
    while (i < n1) {
        array[k] = L[i];
        i++;
        k++;
    }
    while (j < n2) {
        array[k] = R[j];
        j++;
        k++;
    }
    // メモリを解放
    free(L);
    free(R);
}

// マージソートの完全な実装
void mergeSort(int *array, int left, int right) {
    if (left < right) {
        int middle = left + (right - left) / 2;
        mergeSort(array, left, middle);
        mergeSort(array, middle + 1, right);
        merge(array, left, middle, right);
    }
}
int main() {
    int array[] = {12, 11, 13, 5, 6, 7};
    int array_size = sizeof(array) / sizeof(array[0]);
    printf("Given array is \n");
    for (int i = 0; i < array_size; i++) printf("%d ", array[i]);
    printf("\n");
    mergeSort(array, 0, array_size - 1);
    printf("\nSorted array is \n");
    for (int i = 0; i < array_size; i++) printf("%d ", array[i]);
    printf("\n");
    return 0;
}
Given array is 
12 11 13 5 6 7 
Sorted array is 
5 6 7 11 12 13 

このプログラムは、与えられた配列をマージソートを用いて昇順にソートします。

分割と統合の処理を組み合わせることで、効率的にソートを行うことができます。

ポインタを用いたマージソートの応用例

ポインタを用いたマージソートは、さまざまな応用が可能です。

ここでは、いくつかの応用例を紹介します。

動的メモリを用いたマージソート

動的メモリを用いることで、配列のサイズを実行時に決定することができます。

これにより、入力データのサイズに応じて柔軟にメモリを確保し、マージソートを実行することが可能です。

#include <stdio.h>
#include <stdlib.h>
// 動的メモリを用いたマージソートの例
int main() {
    int n;
    printf("Enter number of elements: ");
    scanf("%d", &n);
    int *array = (int *)malloc(n * sizeof(int));
    if (array == NULL) {
        fprintf(stderr, "Memory allocation failed\n");
        return 1;
    }
    printf("Enter %d integers: ", n);
    for (int i = 0; i < n; i++) {
        scanf("%d", &array[i]);
    }
    mergeSort(array, 0, n - 1);
    printf("Sorted array: ");
    for (int i = 0; i < n; i++) {
        printf("%d ", array[i]);
    }
    printf("\n");
    free(array);
    return 0;
}

このプログラムでは、ユーザーから配列のサイズと要素を入力してもらい、動的にメモリを確保してマージソートを実行します。

リンクリストへの応用

リンクリストは、配列とは異なり、要素が連続していないデータ構造です。

ポインタを用いることで、リンクリスト上でもマージソートを実装することができます。

#include <stdio.h>
#include <stdlib.h>
// リンクリストのノード
struct Node {
    int data;
    struct Node* next;
};
// リンクリストのマージソートの例
void mergeSortList(struct Node** headRef) {
    // 実装は省略
}

リンクリストに対するマージソートは、ノードを分割し、再帰的にソートしてから統合することで実現されます。

マルチスレッド環境でのマージソート

マルチスレッド環境では、分割された配列を並行してソートすることで、処理速度を向上させることができます。

スレッドを用いることで、各部分を独立して処理することが可能です。

#include <pthread.h>
// スレッドを用いたマージソートの例
void* threadMergeSort(void* arg) {
    // 実装は省略
}

スレッドを用いることで、CPUのコアを有効に活用し、ソートのパフォーマンスを向上させることができます。

大規模データセットへの適用

大規模なデータセットに対しても、ポインタを用いたマージソートは有効です。

特に、外部メモリを使用する場合や、データがメモリに収まりきらない場合に適しています。

  • 外部ソート: データを部分的にメモリに読み込み、ソートしてから統合する。
  • 分散処理: データを複数のマシンに分散し、並行してソートを行う。

これらの手法を用いることで、大規模データセットに対しても効率的にマージソートを適用することができます。

まとめ

ポインタを用いたマージソートは、効率的なメモリ管理と安定したソート性能を提供します。

この記事では、ポインタを活用したマージソートの実装方法とその応用例について解説しました。

これを機に、ポインタを活用したプログラミングに挑戦し、より高度なアルゴリズムの実装に取り組んでみてください。

関連記事

Back to top button