亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

PHP設(shè)計(jì)模式之迭代器模式的使用

 更新時(shí)間:2021年04月29日 09:03:09   作者:硬核項(xiàng)目經(jīng)理  
這篇文章主要介紹了PHP設(shè)計(jì)模式之迭代器模式的使用,幫助大家更好的理解和學(xué)習(xí)使用PHP,感興趣的朋友可以了解下

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

Gof類圖及解釋

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

代碼實(shí)現(xiàn)

interface Aggregate
{
    public function CreateIterator();
}

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

首先是聚合類,也就是可以進(jìn)行迭代的類,這里因?yàn)槲沂敲嫦驅(qū)ο蟮脑O(shè)計(jì)模式,所以迭代器模式針對的是對一個(gè)類的內(nèi)容進(jìn)行迭代。在這里,其實(shí)我們也只是模擬了一個(gè)數(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];
    }
}

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

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

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

客戶端直接使用while來進(jìn)行操作即可。

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

我們的手機(jī)工廠不得了,自己組裝了一條生產(chǎn)線,這條生產(chǎn)線主要是做什么的呢?成型機(jī)我們已經(jīng)交給富X康來搞定了,我們這條線就是給手機(jī)刷顏色的。當(dāng)我們把所有已經(jīng)交貨的手機(jī)(Aggregate)放到不同的生產(chǎn)線后(Iterator),就會(huì)一臺(tái)一臺(tái)的幫我們刷上當(dāng)前生產(chǎn)線的顏色,是不是很強(qiáng)大!!科技不止于換殼,這條線還在,我們就可以再做別的事兒,比如加點(diǎn)掛繩什么的,反正只要能一臺(tái)一臺(tái)的通過我就能裝上東西,你說好用不好用?。?/p>

完整代碼:github.com/zhangyue050…

實(shí)例

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

消息發(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)
    {
        // 反轉(zhuǎn)數(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ā)的短信號(hào)碼列表
$mobileList = [
    '13111111111',
    '13111111112',
    '13111111113',
    '13111111114',
    '13111111115',
    '13111111116',
    '13111111117',
    '13111111118',
];

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

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

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

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

說明

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

完整源碼:github.com/zhangyue050…

彩蛋

PHP中的Iterator接口已經(jīng)為我們準(zhǔn)備好了一套標(biāo)準(zhǔn)的Iterator模式的實(shí)現(xiàn),而且(這里需要畫重點(diǎn)),實(shí)現(xiàn)這個(gè)接口的類可以用foreach來遍歷哦!

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

源碼:github.com/zhangyue050…

文檔中相關(guān)的接口都可以看看,更重要的是,PHP的SPL擴(kuò)展中,也為我們準(zhǔn)備了很多常用的迭代器封裝。要知道,面試的時(shí)候要是能說出這里面的幾個(gè)來,那面試官可是也會(huì)刮目相看的哦!

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

以上就是PHP設(shè)計(jì)模式之迭代器模式的使用的詳細(xì)內(nèi)容,更多關(guān)于PHP 迭代器模式的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • php如何獲取文件的擴(kuò)展名

    php如何獲取文件的擴(kuò)展名

    這篇文章主要介紹了php如何獲取文件的擴(kuò)展名,即文件后綴名的方法做一個(gè)總結(jié),感興趣的小伙伴們可以參考一下。
    2015-10-10
  • php過濾器使用詳解

    php過濾器使用詳解

    PHP 過濾器用于驗(yàn)證和清理外部輸入。PHP 過濾器擴(kuò)展具有檢查用戶輸入所需的許多功能,旨在使數(shù)據(jù)驗(yàn)證更容易、更快
    2022-11-11
  • php IP轉(zhuǎn)換整形(ip2long)的詳解

    php IP轉(zhuǎn)換整形(ip2long)的詳解

    本篇文章是對php中IP轉(zhuǎn)換整形(ip2long)進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-06-06
  • highchart數(shù)據(jù)源縱軸json內(nèi)的值必須是int(詳解)

    highchart數(shù)據(jù)源縱軸json內(nèi)的值必須是int(詳解)

    下面小編就為大家?guī)硪黄猦ighchart數(shù)據(jù)源縱軸json內(nèi)的值必須是int(詳解)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-02-02
  • 解析PHP跳出循環(huán)的方法以及continue、break、exit的區(qū)別介紹

    解析PHP跳出循環(huán)的方法以及continue、break、exit的區(qū)別介紹

    本篇文章是對PHP跳出循環(huán)的方法以及continue、break、exit的區(qū)別進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-07-07
  • php中通過數(shù)組進(jìn)行高效隨機(jī)抽取指定條記錄的算法

    php中通過數(shù)組進(jìn)行高效隨機(jī)抽取指定條記錄的算法

    php使用數(shù)組array_rand()函數(shù)進(jìn)行高效隨機(jī)抽取指定條數(shù)的記錄,可以隨機(jī)抽取數(shù)據(jù)庫中的記錄,適合進(jìn)行隨機(jī)展示和抽獎(jiǎng)程序
    2013-09-09
  • PHP unset函數(shù)原理及使用方法解析

    PHP unset函數(shù)原理及使用方法解析

    這篇文章主要介紹了PHP unset函數(shù)原理及使用方法解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-08-08
  • PHP超級(jí)全局變量數(shù)組小結(jié)

    PHP超級(jí)全局變量數(shù)組小結(jié)

    PHP超級(jí)全局變量數(shù)組(Super Global Array),又稱為PHP預(yù)定義數(shù)組,是由PHP引擎內(nèi)置的,不需要開發(fā)者重新定義。 在PHP腳本運(yùn)行時(shí),PHP會(huì)自動(dòng)將一些數(shù)據(jù)放在超級(jí)全局?jǐn)?shù)組中
    2012-10-10
  • PHP進(jìn)程通信基礎(chǔ)之信號(hào)

    PHP進(jìn)程通信基礎(chǔ)之信號(hào)

    這篇文章主要介紹了PHP進(jìn)程通信基礎(chǔ)中的信號(hào)相關(guān)的基礎(chǔ)知識(shí)及示例,非常不錯(cuò),有需要的小伙伴可以參考下
    2017-02-02
  • php array_flip() 刪除數(shù)組重復(fù)元素

    php array_flip() 刪除數(shù)組重復(fù)元素

    在PHP中,用于刪除數(shù)組中重復(fù)元素有一個(gè)可用的函數(shù),那就是 array_unique(), 但是它并不是一個(gè)最高效的方法,使用array_flip() 函數(shù)將比array_uniqure()在速度上高出五倍左右。
    2009-01-01

最新評(píng)論