前の記事「PHP のオブザーバー パターンの詳細分析」では、PHP のオブザーバー パターンについて紹介しましたが、この記事では、PHP のイテレーター パターンについて理解します。
このパターンについて話すときは、ループ ステートメントについて言及する必要があります。 『Dahua Design Pattern』の中で著者は、このモデルの学習上の重要性が実践上の重要性を上回っていると述べていますが、これはなぜでしょうか。もちろん、それはforeachのせいでした。どの言語にも、配列やオブジェクトを迅速かつ簡単に走査できるこの種の類似した構文があり、そのためイテレータ パターンが 23 の主要なデザイン パターンの中ではスターから徐々に脇役になっていきます。特に PHP 言語の場合、PHP の強みは配列の柔軟な操作にあります。これはハッシュマップ構造です。当然のことながら、さまざまな便利な配列操作構文があり、foreach は for よりもさらに優れた最も一般的に使用されるステートメントでもあります。使用済み。
GoF 定義: 集合オブジェクト内の各要素を公開せずに順次アクセスするメソッドを提供します。オブジェクトの
#GoF クラス図
##コード実装
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 を直接使用して操作できます。
うちの携帯電話工場はすごいんですが、自社で生産ラインを組み立てているんですが、この生産ラインは主に何に使われているんですか?成形機はフジコン社に引き渡し、当社のラインは携帯電話の塗装に使用されています。納品されたすべての携帯電話(Aggregate)を別の生産ライン(Iterator)に投入すると、現在の生産ラインの色を一色ずつ染めていくのに役立ちます。 !技術というのは貝殻を変えるだけではありません。このラインが残っているなら、ランヤードを追加したり、他のこともできます。とにかく、一つ一つ通過できれば、何かを着ることはできます。できると思いますか?使いやすい! !
完全なコード: https://github.com/zhangyue0503/designpatterns-php/blob/master/07.iterator/source/iterator.php
实例还是围绕着我们的短信发送来看。这一次,我们的业务需求是尽快的发一批通知短信给用户,因为活动的时候可不等人啊。在之前我们会使用多个脚本来把用户手机号分成多组来进行发送。现在我们可以用swoole来直接多线程的发送。所要达到的效果其实就是为了快速的把成百上千的短信发完。这个时候我们也会做一些策略,比如数据库里是100条要送的短信,有个字段是发送状态,一个线程正序的发,一个线程倒序的发,当正序和倒序都发送到50条的时候其实已经同步的发完这100条了,不过也有可能会有失败的情况出现,这时,两个线程还会继续去发送那些上次发送不成功的信息,这样能够最大程度的确保发送的效率和到达率。
消息发送迭代器类图
完整源码: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 = [ '13111111111', '13111111112', '13111111113', '13111111114', '13111111115', '13111111116', '13111111117', '13111111118', ]; // 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 サイトの他の関連記事を参照してください。