您好,登錄后才能下訂單哦!
小編給大家分享一下PHP中中迭代器模式是什么,希望大家閱讀完這篇文章之后都有所收獲,下面讓我們一起去探討吧!
一說到這個模式,就不得不提循環(huán)語句。在《大話設計模式》中,作者說道這個模式現(xiàn)在的學習意義更大于實際意義,這是為什么呢?當然就是被foreach這貨給整得。任何語言都有這種類似的語法可以方便快捷的對數(shù)組、對象進行遍歷,從而讓迭代器模式從高高在上的23大設計模式中的明星慢慢成為了路人。特別是我們這門PHP語言,PHP的強大之處就在于對于數(shù)組的靈活操作,本身就是hashmap的結(jié)構,自然會有各種方便的數(shù)組操作語法,而foreach也是我們最常用的語句,甚至比for還常用。
GoF定義:提供一種方法順序訪問一個聚合對象中各個元素,而又不需暴露該對象的內(nèi)部表示
GoF類圖
代碼實現(xiàn)
interface Aggregate { public function CreateIterator(); } class ConcreteAggregate implements Aggregate { public function CreateIterator() { $list = [ "a", "b", "c", "d", ]; return new ConcreteIterator($list); } }
首先是聚合類,也就是可以進行迭代的類,這里因為我是面向?qū)ο蟮脑O計模式,所以迭代器模式針對的是對一個類的內(nèi)容進行迭代。在這里,其實我們也只是模擬了一個數(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ù)結(jié)構或數(shù)據(jù)庫時對游標進行的操作。用First()和Next()來移動游標,用CurrentItem()來獲得當前游標的數(shù)據(jù)內(nèi)容,用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ù)組內(nèi)容就可以這樣存在一個Bean中。這時,我們使用迭代器就可以只針對自己這個對象內(nèi)部的這個數(shù)組來進行操作啦!
上述Java的內(nèi)容其實是筆者在做Android開發(fā)時經(jīng)常會用到的,有時數(shù)據(jù)庫的JavaBean也會出現(xiàn)這種數(shù)組來存儲外鍵。但在PHP中一般很少使用,因為PHP中大部分的AR對象和Java中的Bean概念還是略有不同。有興趣的同學可以了解下!
我們的手機工廠不得了,自己組裝了一條生產(chǎn)線,這條生產(chǎn)線主要是做什么的呢?成型機我們已經(jīng)交給富X康來搞定了,我們這條線就是給手機刷顏色的。當我們把所有已經(jīng)交貨的手機(Aggregate)放到不同的生產(chǎn)線后(Iterator),就會一臺一臺的幫我們刷上當前生產(chǎn)線的顏色,是不是很強大!!科技不止于換殼,這條線還在,我們就可以再做別的事兒,比如加點掛繩什么的,反正只要能一臺一臺的通過我就能裝上東西,你說好用不好用!!
完整代碼:https://github.com/zhangyue0503/designpatterns-php/blob/master/07.iterator/source/iterator.php
實例還是圍繞著我們的短信發(fā)送來看。這一次,我們的業(yè)務需求是盡快的發(fā)一批通知短信給用戶,因為活動的時候可不等人啊。在之前我們會使用多個腳本來把用戶手機號分成多組來進行發(fā)送?,F(xiàn)在我們可以用swoole來直接多線程的發(fā)送。所要達到的效果其實就是為了快速的把成百上千的短信發(fā)完。這個時候我們也會做一些策略,比如數(shù)據(jù)庫里是100條要送的短信,有個字段是發(fā)送狀態(tài),一個線程正序的發(fā),一個線程倒序的發(fā),當正序和倒序都發(fā)送到50條的時候其實已經(jīng)同步的發(fā)完這100條了,不過也有可能會有失敗的情況出現(xiàn),這時,兩個線程還會繼續(xù)去發(fā)送那些上次發(fā)送不成功的信息,這樣能夠最大程度的確保發(fā)送的效率和到達率。
消息發(fā)送迭代器類圖
完整源碼: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) { // 反轉(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ā)的短信號碼列表 $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ù)組的操作,另外用兩個類似于工廠方法模式的類來對迭代器進行封裝
例子非常簡單,但有時候這種用法也非常實用,比如一些搜索引擎排名的爬蟲,多次確認某些關鍵詞的排名,這時候我們就可以正著、反著來回進行驗證
看完了這篇文章,相信你對“PHP中中迭代器模式是什么”有了一定的了解,如果想了解更多相關知識,歡迎關注億速云行業(yè)資訊頻道,感謝各位的閱讀!
免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權內(nèi)容。