ホームページ > バックエンド開発 > PHPの問題 > PHP のイテレータ パターンの簡単な説明

PHP のイテレータ パターンの簡単な説明

青灯夜游
リリース: 2023-03-11 07:12:01
転載
1567 人が閲覧しました

前の記事「PHP のオブザーバー パターンの詳細分析」では、PHP のオブザーバー パターンについて紹介しましたが、この記事では、PHP のイテレーター パターンについて理解します。

PHP のイテレータ パターンの簡単な説明

このパターンについて話すときは、ループ ステートメントについて言及する必要があります。 『Dahua Design Pattern』の中で著者は、このモデルの学習上の重要性が実践上の重要性を上回っていると述べていますが、これはなぜでしょうか。もちろん、それはforeachのせいでした。どの言語にも、配列やオブジェクトを迅速かつ簡単に走査できるこの種の類似した構文があり、そのためイテレータ パターンが 23 の主要なデザイン パターンの中ではスターから徐々に脇役になっていきます。特に PHP 言語の場合、PHP の強みは配列の柔軟な操作にあります。これはハッシュマップ構造です。当然のことながら、さまざまな便利な配列操作構文があり、foreach は for よりもさらに優れた最も一般的に使用されるステートメントでもあります。使用済み。

Gof クラス図と説明

GoF 定義: 集合オブジェクト内の各要素を公開せずに順次アクセスするメソッドを提供します。オブジェクトの

#GoF クラス図

PHP のイテレータ パターンの簡単な説明

##コード実装

interface Aggregate
{
    public function CreateIterator();
}

class ConcreteAggregate implements Aggregate
{
    public function CreateIterator()
    {
        $list = [
            "a",
            "b",
            "c",
            "d",
        ];
        return new ConcreteIterator($list);
    }
}
ログイン後にコピー

1 つ目は、反復可能なクラスである集約クラスです。私はオブジェクト指向設計パターンなので、反復子パターンはクラスの内容を反復することを目的としています。実際、ここでは配列をシミュレートしてそれを反復子に渡しただけです。

interface MyIterator
{
    public function First();
    public function Next();
    public function IsDone();
    public function CurrentItem();
}

class ConcreteIterator implements MyIterator
{
    private $list;
    private $index;
    public function __construct($list)
    {
        $this->list = $list;
        $this->index = 0;
    }
    public function First()
    {
        $this->index = 0;
    }

    public function Next()
    {
        $this->index++;
    }

    public function IsDone()
    {
        return $this->index >= count($this->list);
    }

    public function CurrentItem()
    {
        return $this->list[$this->index];
    }
}
ログイン後にコピー

イテレータはデビューし、主にコレクション データを操作する 4 つのメソッドを実装します。データ構造やデータベースを学習するときにカーソルを使って行うことと少し似ています。 First()とNext()でカーソルの移動、CurrentItem()で現在のカーソルのデータ内容を取得、IsDone()で次のデータがあるか確認します。したがって、このモードは

カーソル モード とも呼ばれます。

$agreegate = new ConcreteAggregate();
$iterator = $agreegate->CreateIterator();

while (!$iterator->IsDone()) {
    echo $iterator->CurrentItem(), PHP_EOL;
    $iterator->Next();
}
ログイン後にコピー

クライアントは while を直接使用して操作できます。

    誰もが不思議に思っているはずですが、なぜイテレータ インターフェイス クラスの名前は Iterator ではないのでしょうか?試してみると、PHP がこのインターフェースを用意してくれていることがわかり、実装後は、foreach を使用して、Iterator インターフェースを実装したこのクラスを利用することができます。最後にこのクラスの使い方を見てみましょう。
  • クラスを横断することに同意しませんでしたか?なぜ配列をやり取りするのでしょうか? Java を開発したことのある学生は、Object クラスという名前の JavaBean 内に、List myList などの List 型の変数が現在のオブジェクトのコレクションを表すように記述されていることを知っている必要があります。使用中にこの List にデータを追加した後、次回は Object.myList を直接使用してデータのセットを取得できます。たとえば、インターフェースから取得した json 配列の内容を Bean に格納できます。現時点では、イテレータを使用して、独自のオブジェクト内の配列のみを操作できます。
  • 上記の Java コンテンツは、実際に著者が Android 開発を行うときによく使用するものであり、データベースの JavaBean にも外部キーを格納するためにこの種の配列が含まれる場合があります。ただし、PHP のほとんどの AR オブジェクトは Java の Bean の概念と若干異なるため、一般に PHP で使用されることはほとんどありません。興味のある学生はさらに詳しく学ぶことができます!

うちの携帯電話工場はすごいんですが、自社で生産ラインを組み立てているんですが、この生産ラインは主に何に使われているんですか?成形機はフジコン社に引き渡し、当社のラインは携帯電話の塗装に使用されています。納品されたすべての携帯電話(Aggregate)を別の生産ライン(Iterator)に投入すると、現在の生産ラインの色を一色ずつ染めていくのに役立ちます。 !技術というのは貝殻を変えるだけではありません。このラインが残っているなら、ランヤードを追加したり、他のこともできます。とにかく、一つ一つ通過できれば、何かを着ることはできます。できると思いますか?使いやすい! !

完全なコード: https://github.com/zhangyue0503/designpatterns-php/blob/master/07.iterator/source/iterator.php

实例

实例还是围绕着我们的短信发送来看。这一次,我们的业务需求是尽快的发一批通知短信给用户,因为活动的时候可不等人啊。在之前我们会使用多个脚本来把用户手机号分成多组来进行发送。现在我们可以用swoole来直接多线程的发送。所要达到的效果其实就是为了快速的把成百上千的短信发完。这个时候我们也会做一些策略,比如数据库里是100条要送的短信,有个字段是发送状态,一个线程正序的发,一个线程倒序的发,当正序和倒序都发送到50条的时候其实已经同步的发完这100条了,不过也有可能会有失败的情况出现,这时,两个线程还会继续去发送那些上次发送不成功的信息,这样能够最大程度的确保发送的效率和到达率。

消息发送迭代器类图

PHP のイテレータ パターンの簡単な説明

完整源码:https://github.com/zhangyue0503/designpatterns-php/blob/master/07.iterator/source/iterator-msg.php

<?php

interface MsgIterator
{
    public function First();
    public function Next();
    public function IsDone();
    public function CurrentItem();
}

// 正向迭代器
class MsgIteratorAsc implements MsgIterator
{
    private $list;
    private $index;
    public function __construct($list)
    {
        $this->list = $list;
        $this->index = 0;
    }
    public function First()
    {
        $this->index = 0;
    }

    public function Next()
    {
        $this->index++;
    }

    public function IsDone()
    {
        return $this->index >= count($this->list);
    }

    public function CurrentItem()
    {
        return $this->list[$this->index];
    }
}

// 反向迭代器
class MsgIteratorDesc implements MsgIterator
{
    private $list;
    private $index;
    public function __construct($list)
    {
        // 反转数组
        $this->list = array_reverse($list);
        $this->index = 0;
    }
    public function First()
    {
        $this->index = 0;
    }

    public function Next()
    {
        $this->index++;
    }

    public function IsDone()
    {
        return $this->index >= count($this->list);
    }

    public function CurrentItem()
    {
        return $this->list[$this->index];
    }
}

interface Message
{
    public function CreateIterator($list);
}

class MessageAsc implements Message
{
    public function CreateIterator($list)
    {
        return new MsgIteratorAsc($list);
    }
}
class MessageDesc implements Message
{
    public function CreateIterator($list)
    {
        return new MsgIteratorDesc($list);
    }
}

// 要发的短信号码列表
$mobileList = [
    &#39;13111111111&#39;,
    &#39;13111111112&#39;,
    &#39;13111111113&#39;,
    &#39;13111111114&#39;,
    &#39;13111111115&#39;,
    &#39;13111111116&#39;,
    &#39;13111111117&#39;,
    &#39;13111111118&#39;,
];

// A服务器脚本或使用swoole发送正向的一半
$serverA = new MessageAsc();
$iteratorA = $serverA->CreateIterator($mobileList);

while (!$iteratorA->IsDone()) {
    echo $iteratorA->CurrentItem(), PHP_EOL;
    $iteratorA->Next();
}

// B服务器脚本或使用swoole同步发送反向的一半
$serverB = new MessageDesc();
$iteratorB = $serverB->CreateIterator($mobileList);

while (!$iteratorB->IsDone()) {
    echo $iteratorB->CurrentItem(), PHP_EOL;
    $iteratorB->Next();
}
ログイン後にコピー

说明

  • 其实就是两个迭代器,一个是正序一个是倒序,然后遍历数组
  • 例子中我们还是对一个数组的操作,另外用两个类似于工厂方法模式的类来对迭代器进行封装
  • 例子非常简单,但有时候这种用法也非常实用,比如一些搜索引擎排名的爬虫,多次确认某些关键词的排名,这时候我们就可以正着、反着来回进行验证

完整源码:https://github.com/zhangyue0503/designpatterns-php/blob/master/06.observer/source/spl_observer.php

彩蛋

PHP中的Iterator接口已经为我们准备好了一套标准的Iterator模式的实现,而且(这里需要画重点),实现这个接口的类可以用foreach来遍历哦!

文档:https://www.php.net/manual/zh/class.iterator.php

源码:https://github.com/zhangyue0503/designpatterns-php/blob/master/07.iterator/source/iterator-php.php

文档中相关的接口都可以看看,更重要的是,PHP的SPL扩展中,也为我们准备了很多常用的迭代器封装。要知道,面试的时候要是能说出这里面的几个来,那面试官可是也会刮目相看的哦!

SPL迭代器:https://www.php.net/manual/zh/spl.iterators.php

本文转载自:https://juejin.cn/post/6844903937921777671

作者:硬核项目经理

推荐学习:《PHP视频教程

以上がPHP のイテレータ パターンの簡単な説明の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

ソース:juejin.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新の問題
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート