<一覧に戻る

Big-O記法の理解と適用例

Big-O記法は、アルゴリズムの計算量を表現するための重要なツールです。計算量理論では、主に時間計算量と空間計算量が考慮されます。ここでは、Big-O記法の基本的な概念と、Pythonでの実装例を通じて理解を深めていきます。

Big-O記法の基本概念

Big-O記法は、アルゴリズムの最悪の場合の時間(または空間)計算量を、入力サイズに対する関数として表現します。これは、入力が大きくなるときにアルゴリズムの性能がどのように変化するかを示します。

主なBig-O記法の例

  • O(1): 定数時間 - 入力のサイズに関係なく、実行時間が一定。
  • O(log n): 対数時間 - 入力サイズが増加するにつれて、実行時間がわずかに増加。
  • O(n): 線形時間 - 入力サイズが1増えるごとに、実行時間も1増える。
  • O(n log n): 線形対数時間 - よくあるソートアルゴリズム(例: マージソート)。
  • O(n^2): 二次時間 - 入力サイズが増えるごとに、実行時間が平方に増加(例: バブルソート)。
  • O(2^n): 指数時間 - 入力サイズが1増えるごとに、実行時間が指数的に増加(例: フィボナッチ数列の再帰的計算)。

サンプルコード

以下に、いくつかのアルゴリズムの実装例を示します。それぞれのアルゴリズムに対するBig-O記法を考察します。

1. 定数時間O(1)

def get_first_element(arr):
    return arr[0]

# 実行例
sample_array = [10, 20, 30, 40]
print(get_first_element(sample_array))  # 出力: 10

この関数は、配列の最初の要素を返すだけなので、入力のサイズに関係なく常に一定の時間で実行されます。

2. 線形時間O(n)

def print_all_elements(arr):
    for element in arr:
        print(element)

# 実行例
sample_array = [10, 20, 30, 40]
print_all_elements(sample_array)

この関数は配列のすべての要素を出力します。したがって、配列のサイズnに対してO(n)の時間計算量を持ちます。

3. 二次時間O(n^2)

def bubble_sort(arr):
    n = len(arr)
    for i in range(n):
        for j in range(0, n-i-1):
            if arr[j] > arr[j+1]:
                arr[j], arr[j+1] = arr[j+1], arr[j]
    return arr

# 実行例
sample_array = [64, 34, 25, 12, 22, 11, 90]
sorted_array = bubble_sort(sample_array)
print(sorted_array)  # 出力: [11, 12, 22, 25, 34, 64, 90]

バブルソートは、配列のすべての要素を比較するため、最悪の場合にはO(n^2)の時間計算量になります。

4. 線形対数時間O(n log n)

def merge_sort(arr):
    if len(arr) > 1:
        mid = len(arr) // 2
        left_half = arr[:mid]
        right_half = arr[mid:]

        merge_sort(left_half)
        merge_sort(right_half)

        i = j = k = 0

        while i < len(left_half) and j < len(right_half):
            if left_half[i] < right_half[j]:
                arr[k] = left_half[i]
                i += 1
            else:
                arr[k] = right_half[j]
                j += 1
            k += 1

        while i < len(left_half):
            arr[k] = left_half[i]
            i += 1
            k += 1

        while j < len(right_half):
            arr[k] = right_half[j]
            j += 1
            k += 1

# 実行例
sample_array = [38, 27, 43, 3, 9, 82, 10]
merge_sort(sample_array)
print(sample_array)  # 出力: [3, 9, 10, 27, 38, 43, 82]

マージソートは分割統治法に基づくソートアルゴリズムで、平均的にO(n log n)の時間計算量を持ちます。これは配列を再帰的に分割し、各部分をソートして結合するためです。

まとめ

Big-O記法は、アルゴリズムの効率を理解し、比較するための非常に重要なツールです。異なるアルゴリズムの時間計算量を知っておくことで、適切なアルゴリズムを選択する手助けとなります。今回紹介したサンプルコードを通じて、実際にどのように計算量が変化するかを体験し、理解を深めてください。

出力結果: