Webエンジニア向けプログラミング解説動画をYouTubeで配信中!
▶ チャンネル登録はこちら

【PHP8.x】FilterIterator::valid()メソッドの使い方

validメソッドの使い方について、初心者にもわかりやすく解説します。

作成日: 更新日:

基本的な使い方

validメソッドは、イテレータの現在の要素が有効かどうかをチェックするメソッドです。このメソッドは、FilterIteratorクラス、またはそのサブクラスのインスタンスに対して使用されます。FilterIteratorは、配列や他のイテレータブルなオブジェクトの要素を特定の条件でフィルタリングするために利用される抽象クラスです。foreachループなどでイテレータを反復処理する際、PHPのエンジンは内部的にこのvalidメソッドを呼び出します。validメソッドがtrueを返す条件は2つあります。まず、元となる内部イテレータがまだ終端に達しておらず、有効な要素を指している必要があります。次に、その要素がFilterIteratorのサブクラスで実装されたacceptメソッドのフィルタリング条件を満たしている必要があります。この両方の条件が満たされた場合にのみtrueを返し、そうでなければfalseを返します。validメソッドがfalseを返すと、反復処理は終了します。したがって、このメソッドはフィルタリング処理において、ループを継続するかどうかを決定する重要な役割を担っています。

構文(syntax)

1<?php
2
3// 偶数のみを通過させるカスタムイテレータ
4class EvenNumberFilter extends FilterIterator
5{
6    public function accept(): bool
7    {
8        // 親イテレータの現在の要素が偶数なら true を返す
9        return $this->current() % 2 === 0;
10    }
11}
12
13// フィルタリングする元のデータを持つイテレータ
14$numbers = new ArrayIterator([1, 2, 3, 4, 5, 6, 7, 8]);
15
16// カスタムイテレータをインスタンス化
17$iterator = new EvenNumberFilter($numbers);
18
19// valid() は、イテレータの現在の位置が有効かどうかをチェックします。
20// while ループは、valid() が true を返す間、処理を続けます。
21while ($iterator->valid()) {
22    // 現在のキーと値を取得
23    $key = $iterator->key();
24    $value = $iterator->current();
25    echo "{$key}: {$value}" . PHP_EOL;
26
27    // 次の要素に進める
28    $iterator->next();
29}
30
31?>

引数(parameters)

引数なし

引数はありません

戻り値(return)

bool

現在のイテレータが有効な要素を指している場合に true を返します。無効な場合は false を返します。

サンプルコード

PHP FilterIteratorでのバリデーション

1<?php
2
3/**
4 * 特定の条件を満たす要素のみを通過させるフィルタクラス。
5 * この例では、偶数のみを有効なデータとして検証(バリデーション)します。
6 */
7class EvenNumberFilter extends FilterIterator
8{
9    /**
10     * 現在の要素がイテレーションで有効かどうかを判断します。
11     * FilterIteratorでは、このaccept()メソッドが実質的なバリデーションルールを定義します。
12     *
13     * @return bool 現在の要素が偶数の場合は true、そうでない場合は false を返します。
14     */
15    public function accept(): bool
16    {
17        // current() で現在の要素の値を取得し、それが偶数かどうかを検証します。
18        $currentValue = $this->current();
19        return is_int($currentValue) && $currentValue % 2 === 0;
20    }
21}
22
23// 1. 元となるデータ配列
24$numbers = [1, 2, 'text', 3, 4, 5, 6, 7, 8, '9', 10];
25
26// 2. 配列をイテレータに変換
27$arrayIterator = new ArrayIterator($numbers);
28
29// 3. カスタムフィルタを適用したイテレータを作成
30$filterIterator = new EvenNumberFilter($arrayIterator);
31
32// 4. ループ処理
33// FilterIterator::valid() は、フィルタリング後の結果として
34// 有効な要素がまだ存在するかどうかをチェックします。
35// accept() が true を返す要素がなくなるまで、ループは続行されます。
36echo "フィルタリング(検証)後の有効な数値(偶数):" . PHP_EOL;
37foreach ($filterIterator as $key => $value) {
38    echo "[{$key}] => {$value}" . PHP_EOL;
39}

PHPのFilterIterator::valid()メソッドは、フィルタリング後のイテレータに、処理すべき有効な要素がまだ残っているかどうかを判定するために使われます。主にforeachループなどでPHPの内部的に呼び出され、ループを続けるか終了するかを決定する役割を持ちます。

このメソッドに引数はありません。戻り値は真偽値(bool)です。イテレータ内にフィルタリング条件を満たす有効な要素がまだ存在する場合はtrueを、もう存在しない場合はfalseを返します。

サンプルコードでは、配列から偶数のみを抽出(検証)するためのカスタムフィルタEvenNumberFilterを定義しています。具体的な検証ルールはaccept()メソッドに実装されており、現在の要素が整数かつ偶数の場合にtrueを返します。foreachループでこのフィルタ済みイテレータを処理する際、内部でvalid()が呼び出されます。valid()accept()trueを返す要素が見つかるまでイテレータを次に進め、見つかればtrueを返してループ処理を続行します。全ての要素を確認し、有効な要素がなくなるとfalseを返し、ループが終了します。このようにvalid()は、定義した検証ルールに従って、有効なデータだけを効率的に処理するために不可欠なメソッドです。

foreachループでこのイテレータを利用する際、実際の検証ルールはaccept()メソッドに実装します。一方、valid()メソッドは、accept()で選別された有効な要素がまだ残っているかをforeachが内部的に判断するために使われ、開発者が直接コーディングすることは稀です。この役割の違いを理解することが重要です。また、accept()メソッド内では、is_int()のようにデータ型を事前にチェックすることで、意図しない値によるエラーを防ぎ、より安全なコードになります。この仕組みは、偶数判定だけでなく様々な条件に応用してデータを効率的に抽出できます。

PHP FilterIteratorで偶数のみを検証する

1<?php
2
3declare(strict_types=1);
4
5/**
6 * 偶数のみを有効とするカスタムイテレータ。
7 *
8 * FilterIteratorを継承することで、既存のイテレータ(この例ではArrayIterator)の
9 * 要素を特定の条件でフィルタリング(検証)できます。
10 */
11class EvenNumberValidator extends FilterIterator
12{
13    /**
14     * このメソッドで、現在の要素が有効かどうかを判定します。
15     * true を返した要素のみが、イテレーションの対象となります。
16     *
17     * @return bool 現在の要素が偶数なら true、そうでなければ false
18     */
19    public function accept(): bool
20    {
21        // 親イテレータから現在の値を取得
22        $current = $this->current();
23        
24        // 値が整数かつ偶数であるかを検証
25        return is_int($current) && $current % 2 === 0;
26    }
27}
28
29// 検証対象のデータ
30$numbers = [1, 2, 'text', 4, 5, 6, 8, 9.5, 10];
31
32// 配列をイテレート可能にするためにArrayIteratorを使用
33$iterator = new ArrayIterator($numbers);
34
35// 作成したカスタムフィルタ(バリデータ)を適用
36$evenIterator = new EvenNumberValidator($iterator);
37
38// foreach ループでイテレータを処理
39// ループの各繰り返しで、内部的に valid() メソッドが呼び出され、
40// 有効な要素がまだ存在するか(accept()がtrueを返す要素があるか)がチェックされます。
41// valid() が false を返した時点でループは終了します。
42foreach ($evenIterator as $key => $value) {
43    echo "Key: {$key}, Valid Value: {$value}" . PHP_EOL;
44}
45
46?>

PHPのFilterIterator::valid()メソッドは、イテレータの現在の位置に、フィルタ条件を満たす有効な要素が存在するかどうかを確認します。このメソッドは引数を持たず、戻り値としてbool型(trueまたはfalse)を返します。

foreach文などでイテレータを処理する際、ループの各繰り返しが始まる前にこのvalid()メソッドが内部的に呼び出されます。戻り値がtrueの間はループが継続し、falseになると有効な要素はもう存在しないと判断され、ループが終了します。

サンプルコードでは、配列から偶数のみを取り出すカスタムイテレータEvenNumberValidatorを定義しています。foreachループがこのイテレータを処理する際、valid()メソッドは、accept()メソッドで定義された「整数かつ偶数である」という条件を満たす次の要素を探します。条件に合う要素が見つかればtrueを返し、その要素がループ内で処理されます。全ての要素を確認し、条件に合うものがなくなるとfalseを返し、ループは停止します。このようにvalid()は、フィルタリングを伴う繰り返し処理の制御に不可欠な役割を担っています。

FilterIteratorを継承して独自の検証機能を作る際は、valid()メソッド自体を書き換えるのではなく、accept()メソッドに検証ロジックを実装します。foreachなどでループ処理を行うと、valid()が内部的にaccept()を呼び出し、条件に合う要素が見つかるまでイテレータを進めます。accept()trueを返した要素だけが有効と判断され、ループ処理の対象となります。サンプルコードのようにaccept()内ではis_int()などで型チェックを行うことが重要です。これにより、文字列や浮動小数点数など、想定外のデータ型が原因でエラーが発生するのを防ぎ、安全なコードになります。

関連コンテンツ