佳木斯湛栽影视文化发展公司

主頁 > 知識庫 > PHP設計模式之迭代器模式的使用

PHP設計模式之迭代器模式的使用

熱門標簽:阿里云 Mysql連接數(shù)設置 電子圍欄 Linux服務器 科大訊飛語音識別系統(tǒng) 團購網站 服務器配置 銀行業(yè)務

一說到這個模式,就不得不提循環(huán)語句。在《大話設計模式》中,作者說道這個模式現(xiàn)在的學習意義更大于實際意義,這是為什么呢?當然就是被foreach這貨給整得。任何語言都有這種類似的語法可以方便快捷的對數(shù)組、對象進行遍歷,從而讓迭代器模式從高高在上的23大設計模式中的明星慢慢成為了路人。特別是我們這門PHP語言,PHP的強大之處就在于對于數(shù)組的靈活操作,本身就是hashmap的結構,自然會有各種方便的數(shù)組操作語法,而foreach也是我們最常用的語句,甚至比for還常用。

Gof類圖及解釋

GoF定義:提供一種方法順序訪問一個聚合對象中各個元素,而又不需暴露該對象的內部表示

代碼實現(xiàn)

interface Aggregate
{
    public function CreateIterator();
}

class ConcreteAggregate implements Aggregate
{
    public function CreateIterator()
    {
        $list = [
            "a",
            "b",
            "c",
            "d",
        ];
        return new ConcreteIterator($list);
    }
}

首先是聚合類,也就是可以進行迭代的類,這里因為我是面向對象的設計模式,所以迭代器模式針對的是對一個類的內容進行迭代。在這里,其實我們也只是模擬了一個數(shù)組交給了迭代器。

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];
    }
}

迭代器閃亮登場,主要實現(xiàn)了四個方法來對集合數(shù)據(jù)進行操作。有點像學習數(shù)據(jù)結構或數(shù)據(jù)庫時對游標進行的操作。用First()和Next()來移動游標,用CurrentItem()來獲得當前游標的數(shù)據(jù)內容,用IsDone()來確認是否還有下一條數(shù)據(jù)。所以,這個模式也另稱為游標模式。

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

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

客戶端直接使用while來進行操作即可。

  • 大家一定很好奇,為什么我們的迭代器接口類不用Iterator來命名?試試就知道,PHP為我們準備好了一個這個接口,實現(xiàn)之后就可以用foreach來使用這個實現(xiàn)了Iterator接口的類了,是不是很高大上。我們最后再看這個類的使用。
  • 不是說好對類進行遍歷嗎?為啥來回傳遞一個數(shù)組?開發(fā)過Java的同學一定知道,在一個名為Object類的JavaBean中,會寫一個變量List類型的變量如List myList,用來表示當前對象的集合。在使用的時候給這個List添加數(shù)據(jù)后,下次就可以直接用Object.myList來獲得一組數(shù)據(jù)了。比如從接口中獲得的json數(shù)組內容就可以這樣存在一個Bean中。這時,我們使用迭代器就可以只針對自己這個對象內部的這個數(shù)組來進行操作啦!
  • 上述Java的內容其實是筆者在做Android開發(fā)時經常會用到的,有時數(shù)據(jù)庫的JavaBean也會出現(xiàn)這種數(shù)組來存儲外鍵。但在PHP中一般很少使用,因為PHP中大部分的AR對象和Java中的Bean概念還是略有不同。有興趣的同學可以了解下!

我們的手機工廠不得了,自己組裝了一條生產線,這條生產線主要是做什么的呢?成型機我們已經交給富X康來搞定了,我們這條線就是給手機刷顏色的。當我們把所有已經交貨的手機(Aggregate)放到不同的生產線后(Iterator),就會一臺一臺的幫我們刷上當前生產線的顏色,是不是很強大?。】萍疾恢褂趽Q殼,這條線還在,我們就可以再做別的事兒,比如加點掛繩什么的,反正只要能一臺一臺的通過我就能裝上東西,你說好用不好用?。?/p>

完整代碼:github.com/zhangyue050…

實例

實例還是圍繞著我們的短信發(fā)送來看。這一次,我們的業(yè)務需求是盡快的發(fā)一批通知短信給用戶,因為活動的時候可不等人啊。在之前我們會使用多個腳本來把用戶手機號分成多組來進行發(fā)送?,F(xiàn)在我們可以用swoole來直接多線程的發(fā)送。所要達到的效果其實就是為了快速的把成百上千的短信發(fā)完。這個時候我們也會做一些策略,比如數(shù)據(jù)庫里是100條要送的短信,有個字段是發(fā)送狀態(tài),一個線程正序的發(fā),一個線程倒序的發(fā),當正序和倒序都發(fā)送到50條的時候其實已經同步的發(fā)完這100條了,不過也有可能會有失敗的情況出現(xiàn),這時,兩個線程還會繼續(xù)去發(fā)送那些上次發(fā)送不成功的信息,這樣能夠最大程度的確保發(fā)送的效率和到達率。

消息發(fā)送迭代器類圖

完整源碼:github.com/zhangyue050…

?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)
    {
        // 反轉數(shù)組
        $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);
    }
}

// 要發(fā)的短信號碼列表
$mobileList = [
    '13111111111',
    '13111111112',
    '13111111113',
    '13111111114',
    '13111111115',
    '13111111116',
    '13111111117',
    '13111111118',
];

// A服務器腳本或使用swoole發(fā)送正向的一半
$serverA = new MessageAsc();
$iteratorA = $serverA->CreateIterator($mobileList);

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

// B服務器腳本或使用swoole同步發(fā)送反向的一半
$serverB = new MessageDesc();
$iteratorB = $serverB->CreateIterator($mobileList);

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

說明

  • 其實就是兩個迭代器,一個是正序一個是倒序,然后遍歷數(shù)組
  • 例子中我們還是對一個數(shù)組的操作,另外用兩個類似于工廠方法模式的類來對迭代器進行封裝
  • 例子非常簡單,但有時候這種用法也非常實用,比如一些搜索引擎排名的爬蟲,多次確認某些關鍵詞的排名,這時候我們就可以正著、反著來回進行驗證

完整源碼:github.com/zhangyue050…

彩蛋

PHP中的Iterator接口已經為我們準備好了一套標準的Iterator模式的實現(xiàn),而且(這里需要畫重點),實現(xiàn)這個接口的類可以用foreach來遍歷哦!

文檔:www.php.net/manual/zh/c…

源碼:github.com/zhangyue050…

文檔中相關的接口都可以看看,更重要的是,PHP的SPL擴展中,也為我們準備了很多常用的迭代器封裝。要知道,面試的時候要是能說出這里面的幾個來,那面試官可是也會刮目相看的哦!

SPL迭代器:www.php.net/manual/zh/s…

以上就是PHP設計模式之迭代器模式的使用的詳細內容,更多關于PHP 迭代器模式的資料請關注腳本之家其它相關文章!

您可能感興趣的文章:
  • PHP設計模式(觀察者模式)
  • 淺談PHP設計模式之門面模式Facade
  • 淺談PHP設計模式之對象池模式Pool
  • 詳解PHP設計模式之依賴注入模式
  • 詳解PHP八大設計模式
  • PHP設計模式之原型模式示例詳解
  • PHP設計模式之命令模式示例詳解
  • PHP八大設計模式案例詳解

標簽:江蘇 萍鄉(xiāng) 蚌埠 衢州 廣元 衡水 大理 棗莊

巨人網絡通訊聲明:本文標題《PHP設計模式之迭代器模式的使用》,本文關鍵詞  ;如發(fā)現(xiàn)本文內容存在版權問題,煩請?zhí)峁┫嚓P信息告之我們,我們將及時溝通與處理。本站內容系統(tǒng)采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266
    铜山县| 建宁县| 梁河县| 平塘县| 斗六市| 礼泉县| 贵定县| 浠水县| 兖州市| 齐齐哈尔市| 锡林浩特市| 成安县| 巴中市| 屏东市| 东阳市| 荆门市| 绥芬河市| 西充县| 天祝| 浮山县| 屯昌县| 桂东县| 泗水县| 定远县| 响水县| 扎赉特旗| 乐平市| 海门市| 互助| 伊宁市| 资兴市| 上饶市| 武安市| 河间市| 内黄县| 专栏| 湟源县| 隆化县| 胶南市| 曲阜市| 龙里县|