【PHP8.x】iterator_to_array関数の使い方

iterator_to_array関数の使い方について、初心者にもわかりやすく解説します。

作成日: 更新日:

基本的な使い方

iterator_to_array関数は、指定されたイテレーターからすべての要素を取り出し、新しい配列を作成して返す関数です。

イテレーターとは、データセットの要素を一つずつ順番に処理するための仕組みで、例えばforeachループで繰り返し処理ができるようなオブジェクトのことです。この関数を利用することで、イテレーターが提供するデータを一度にまとめてPHPの配列として扱うことができるようになります。

第一引数には、配列に変換したいイテレーターオブジェクトを指定します。第二引数はオプションで、変換後の配列のキーをどのように扱うかをbool型で指定します。この引数がtrueに設定されている場合、元のイテレーターのキーがそのまま配列のキーとして保持されます。デフォルト値はfalseで、この場合は数値キーが自動的に再割り当てされます。

iterator_to_array関数は、イテレーターの柔軟な反復処理能力と、PHPの豊富な配列操作関数(array_maparray_filterなど)の利便性を組み合わせたい場合に非常に有用です。例えば、データベースの結果セットやファイルから読み込んだデータを、一度に配列として扱って集計や加工を行いたい場合などに活用できます。

ただし、イテレーターが非常に多くの要素を持つ場合、この関数はすべての要素を一度にメモリにロードするため、大量のメモリを消費する可能性があります。そのため、処理するデータ量が多い場合には、メモリ使用量に注意して利用することが重要です。

構文(syntax)

1<?php
2$array = iterator_to_array($iterator, $preserve_keys);
3?>

引数(parameters)

Traversable $iterator, bool $preserve_keys = true

  • Traversable $iterator: 配列に変換したいイテラブルなオブジェクトを指定します。
  • bool $preserve_keys = true: trueの場合、元のイテレータのキーを保持します。falseの場合、キーは再採番されます。

戻り値(return)

array

iterable を配列に変換したものを返します。

サンプルコード

PHPでイテレータを配列に変換する

1<?php
2
3/**
4 * PHPのイテレータを配列に変換する `iterator_to_array` 関数の使用例です。
5 *
6 * この関数は、`Traversable` インターフェースを実装したオブジェクト(イテレータ)を
7 * 通常のPHPの配列に変換します。
8 */
9function convertIteratorToArrayExample(): void
10{
11    // イテレータの元となる連想配列データを用意
12    $data = [
13        'apple' => 10,
14        'banana' => 20,
15        'orange' => 30,
16    ];
17
18    echo "--- 元データ ---\n";
19    print_r($data);
20
21    // ArrayIterator は Traversable インターフェースを実装しているため、
22    // iterator_to_array 関数の引数として使用できます。
23    $iterator = new ArrayIterator($data);
24
25    echo "\n--- iterator_to_array の使用結果 ---\n";
26
27    // ケース1: キーを保持してイテレータを配列に変換する (デフォルト動作)
28    // 第二引数 $preserve_keys が true の場合、元のイテレータのキーが保持されます。
29    // true はデフォルト値なので省略も可能です。
30    echo "[1] キーを保持して配列に変換 (preserve_keys = true):\n";
31    $arrayWithPreservedKeys = iterator_to_array($iterator, true);
32    print_r($arrayWithPreservedKeys);
33    // 出力例:
34    // Array
35    // (
36    //     [apple] => 10
37    //     [banana] => 20
38    //     [orange] => 30
39    // )
40
41    // iterator_to_array はイテレータを最後まで走査します。
42    // 同じイテレータで再度変換を行う場合は、rewind() を呼び出すか
43    // 新しいイテレータを作成する必要があります。ここでは新しいイテレータを作成します。
44    $iterator = new ArrayIterator($data);
45
46    // ケース2: キーを再インデックスしてイテレータを配列に変換する
47    // 第二引数 $preserve_keys が false の場合、キーは0から始まる数値キーに再インデックスされます。
48    echo "\n[2] キーを再インデックスして配列に変換 (preserve_keys = false):\n";
49    $arrayWithoutPreservedKeys = iterator_to_array($iterator, false);
50    print_r($arrayWithoutPreservedKeys);
51    // 出力例:
52    // Array
53    // (
54    //     [0] => 10
55    //     [1] => 20
56    //     [2] => 30
57    // )
58}
59
60// 関数を実行し、動作を確認します。
61convertIteratorToArrayExample();
62
63?>

PHPのiterator_to_array関数は、イテレータと呼ばれる特殊なオブジェクトを通常のPHPの配列に変換するために使用されます。イテレータとは、Traversableインターフェースを実装したオブジェクトのことで、データ集合を順に処理する仕組みを提供します。

この関数は、最初の引数に変換したいイテレータオブジェクトを受け取ります。そして、戻り値として、イテレータから変換された新しい配列を返します。

2番目の引数$preserve_keysは、変換時に元のイテレータのキーを保持するかどうかを決定する真偽値です。 true(デフォルト値)を指定すると、元のイテレータのキーがそのまま配列のキーとして利用されます。サンプルコードの最初の例のように、連想配列のキー('apple', 'banana'など)がそのまま維持されます。 一方、falseを指定すると、配列のキーは0から始まる連続した数値に再インデックスされます。サンプルコードの2番目の例では、キーが0, 1, 2...と数値に振り直されていることが確認できます。

サンプルコードでは、ArrayIteratorという、配列をイテレータとして扱うクラスを使用し、iterator_to_array関数の挙動を具体的に示しています。イテレータは一度最後まで走査されると、再度変換を行う場合は先頭に戻すか、新しいイテレータを作成する必要があるため、2番目の例では新しいArrayIteratorを生成しています。この関数を使うことで、イテレータで提供されるデータを配列として手軽に扱えるようになります。

iterator_to_array関数を使用すると、対象のイテレータは内部的に最後まで処理されます。そのため、一度この関数で変換したイテレータを再度利用しようとしても、中身が空の配列が返ってきてしまいます。もう一度変換したい場合は、サンプルコードのようにイテレータを新しく作り直すか、rewind()メソッドでポインタを先頭に戻す必要があります。第二引数は、省略すると元のキーが維持されますが、falseを明示的に指定するとキーが0から始まる連番に置き換わる点にも注意してください。巨大なデータを扱うイテレータを変換すると、メモリを大量に消費する可能性があるため、本当に配列への変換が必要な場面での利用が推奨されます。

PHP iterator_to_array で再帰ディレクトリを配列化する

1<?php
2
3/**
4 * recursive iterator_to_array の使用例
5 *
6 * このスクリプトは、RecursiveDirectoryIterator と RecursiveIteratorIterator を使用して
7 * ディレクトリを再帰的に走査し、その結果を iterator_to_array 関数で配列に変換する方法を示します。
8 *
9 * 1. テスト用のディレクトリ構造を作成します。
10 * 2. RecursiveDirectoryIterator でディレクトリを再帰的に走査するイテレータを作成します。
11 * 3. RecursiveIteratorIterator でこの再帰イテレータをフラット化し、葉(ファイル)のみを取得します。
12 * 4. iterator_to_array を使用して、イテレータの内容を通常の配列に変換します。
13 * 5. 結果を表示し、作成したテストデータをクリーンアップします。
14 */
15
16// --- 1. テスト用のディレクトリ構造を作成 ---
17$testDir = 'temp_recursive_data';
18$subDir = $testDir . DIRECTORY_SEPARATOR . 'subdir';
19
20if (!is_dir($subDir)) {
21    mkdir($subDir, 0777, true);
22}
23
24file_put_contents($testDir . DIRECTORY_SEPARATOR . 'file1.txt', 'Content of file1');
25file_put_contents($subDir . DIRECTORY_SEPARATOR . 'file2.txt', 'Content of file2');
26file_put_contents($subDir . DIRECTORY_SEPARATOR . 'file3.log', 'Content of file3');
27
28echo "--- テストデータ作成完了 ---\n";
29
30// --- 2. RecursiveDirectoryIterator と RecursiveIteratorIterator を使用して再帰的なイテレータを作成 ---
31// RecursiveDirectoryIterator は指定されたディレクトリのイテレータを作成します。
32// SKIP_DOTS フラグは . と .. をスキップします。
33$directoryIterator = new RecursiveDirectoryIterator(
34    $testDir,
35    RecursiveDirectoryIterator::SKIP_DOTS
36);
37
38// RecursiveIteratorIterator は、RecursiveIterator からの再帰的なイテレーションをフラット化します。
39// LEAVES_ONLY フラグは、イテレーションの「葉」要素(この場合はファイル)のみを返します。
40$recursiveIterator = new RecursiveIteratorIterator(
41    $directoryIterator,
42    RecursiveIteratorIterator::LEAVES_ONLY
43);
44
45echo "--- iterator_to_array を使用して再帰イテレータを配列に変換 --- \n";
46
47// --- 3. iterator_to_array を使用してイテレータの内容を配列に変換 ---
48// 第1引数に Traversable オブジェクト (RecursiveIteratorIterator) を渡します。
49// 第2引数 $preserve_keys はデフォルトで true なので、ここでは省略しています。
50// この場合、ファイルパスが配列の要素となり、数値インデックスが割り当てられます。
51$filesArray = iterator_to_array($recursiveIterator);
52
53// --- 4. 結果を表示 ---
54echo "取得されたファイルリスト:\n";
55foreach ($filesArray as $path) {
56    echo $path . "\n";
57}
58echo "\n";
59
60echo "--- 変換された配列の内容 (var_dump) ---\n";
61var_dump($filesArray);
62echo "\n";
63
64// --- 5. クリーンアップ ---
65function rrmdir(string $dir): void
66{
67    if (!file_exists($dir)) {
68        return;
69    }
70    $objects = scandir($dir);
71    foreach ($objects as $object) {
72        if ($object !== '.' && $object !== '..') {
73            if (is_dir($dir . DIRECTORY_SEPARATOR . $object)) {
74                rrmdir($dir . DIRECTORY_SEPARATOR . $object);
75            } else {
76                unlink($dir . DIRECTORY_SEPARATOR . $object);
77            }
78        }
79    }
80    rmdir($dir);
81}
82
83rrmdir($testDir);
84echo "--- テストデータクリーンアップ完了 ---\n";
85
86?>

iterator_to_arrayはPHP 8.4の拡張機能の一つで、Traversableインターフェースを実装したイテレータオブジェクトの内容を、通常のPHP配列に効率的に変換する関数です。

第一引数$iteratorには、配列に変換したいイテレータ(例: RecursiveIteratorIteratorなど)を指定します。このイテレータは、内部的に要素を順に取得できるオブジェクトである必要があります。 第二引数$preserve_keysはオプションで、デフォルト値はtrueです。この引数をtrueに設定すると、イテレータが現在指している要素のキーがそのまま変換後の配列のキーとして使用されます。falseに設定すると、新しい数値インデックスが0から順に割り当てられます。 戻り値は、イテレータの全ての要素を格納した新しいarrayです。

サンプルコードでは、RecursiveDirectoryIteratorRecursiveIteratorIteratorを組み合わせてディレクトリ内を再帰的に走査し、見つかったファイル(葉要素)のSplFileInfoオブジェクトを列挙するイテレータを作成しています。この再帰イテレータをiterator_to_array関数に渡すことで、取得されたSplFileInfoオブジェクトのリストが通常の配列としてまとめて取得されています。$preserve_keysがデフォルトのtrueであるため、イテレータが生成する数値インデックスとSplFileInfoオブジェクトが、そのまま配列のキーと値として格納されます。

iterator_to_arrayはイテレータの全要素を配列に変換するため、対象データ量が多いとメモリを大量消費する点にご注意ください。第2引数$preserve_keysはイテレータのキーを配列のキーとして保持するかを制御し、デフォルトはtrueです。この場合、イテレータのキー(サンプルではファイルパス)が配列のキーとして設定されます。また、配列の各要素はファイルパスの文字列ではなく、SplFileInfoオブジェクト自体であることにご留意ください。このオブジェクトはファイルの詳細情報を提供し、echoで表示する際にパス文字列へ変換されます。一度iterator_to_arrayを使用するとイテレータは最後まで走査されますので、再度利用する際はrewind()するか、新しいイテレータを生成してください。

関連コンテンツ

【PHP8.x】iterator_to_array関数の使い方 | いっしー@Webエンジニア