一說到這個模式,就不得不提回圈陳述句,在《大話設計模式》中,作者說道這個模式現在的學習意義更大于實際意義,這是為什么呢?當然就是被foreach這貨給整得,任何語言都有這種類似的語法可以方便快捷的對陣列、物件進行遍歷,從而讓迭代器模式從高高在上的23大設計模式中的明星慢慢成為了路人,特別是我們這門PHP語言,PHP的強大之處就在于對于陣列的靈活操作,本身就是hashmap的結構,自然會有各種方便的陣列操作語法,而foreach也是我們最常用的陳述句,甚至比for還常用,
Gof類圖及解釋
GoF定義:提供一種方法順序訪問一個聚合物件中各個元素,而又不需暴露該物件的內部表示
GoF類圖

代碼實作
interface Aggregate
{
public function CreateIterator();
}
class ConcreteAggregate implements Aggregate
{
public function CreateIterator()
{
$list = [
"a",
"b",
"c",
"d",
];
return new ConcreteIterator($list);
}
}
首先是聚合類,也就是可以進行迭代的類,這里因為我是面向物件的設計模式,所以迭代器模式針對的是對一個類的內容進行迭代,在這里,其實我們也只是模擬了一個陣列交給了迭代器,
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];
}
}
迭代器閃亮登場,主要實作了四個方法來對集合資料進行操作,有點像學習資料結構或資料庫時對游標進行的操作,用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
- 上述Java的內容其實是筆者在做Android開發時經常會用到的,有時資料庫的JavaBean也會出現這種陣列來存盤外鍵,但在PHP中一般很少使用,因為PHP中大部分的AR物件和Java中的Bean概念還是略有不同,有興趣的同學可以了解下!
我們的手機工廠不得了,自己組裝了一條生產線,這條生產線主要是做什么的呢?成型機我們已經交給富X康來搞定了,我們這條線就是給手機刷顏色的,當我們把所有已經交貨的手機(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
下期看點
迭代器很好玩吧,而且和觀察者一樣,PHP本身的擴展庫竟然為我們準備了很多介面,平常寫代碼的時候是不是可以炫炫技了呢!!別急,我們進入設計模式的世界并不久,還有很多有意思的設計模式等著我們去學習,就像原型模式,這貨干嘛的?復制自己哦,克隆人的戰爭!
關注公眾號:【硬核專案經理】獲取最新文章
添加微信/QQ好友:【xiaoyuezigonggong/149844827】免費得PHP、專案管理學習資料
知乎、公眾號、抖音、頭條搜索【硬核專案經理】
B站ID:482780532
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/236357.html
標籤:PHP
上一篇:6 JSON
