本文實例講述了PHP生成器(generator)和協(xié)程的實現(xiàn)方法。分享給大家供大家參考,具體如下:
先說一些廢話
PHP 5.5 以來,新的諸多特性又一次令 PHP 煥發(fā)新的光彩,雖然在本文寫的時候已是 PHP 7 alpha 2 發(fā)布后的一段時間,但此時國內(nèi)依舊是 php 5.3 的天下。不過我認(rèn)為新的特性遲早會因為舊的版本的逐漸消失而變得越發(fā)重要,尤其是 PHP 7 的正式版出來后,因此本文的目的就是為了在這之前,幫助一些 PHPer 了解一些他們從沒有了解的東西。所以打算將以本篇作為博客中 PHP 知識補全 系列文章的開篇。
其實在寫本文之前,我對生成器以及基于此特性延伸出來的 php 的協(xié)程實現(xiàn)并沒有比較直觀的了解,主要是我個人水平并不是很高,屬于典型的剛?cè)肓碎T的 PHPer。所以在看了前段時間鳥哥(laruence)博客中對協(xié)程的講解(參考鏈接:《PHP中使用協(xié)同程序?qū)崿F(xiàn)合作多任務(wù)》)后,在我個人對本篇的理解上,針對那些比較難以理解的概念(包括我個人在理解這一概念的時候的難點),以一個更為通俗的方式去講明白。當(dāng)然由于本人也是剛剛?cè)W(xué)習(xí)這一概念,所以有些不得當(dāng)?shù)牡胤皆谒y免,希望大神看見了請不吝賜教。
一切從 Iterator 和 Generator 開始
為便于新入門開發(fā)者理解,本文一半篇幅是講述迭代器接口(Iterator)和 Generator 類的,對此已經(jīng)理解的話,可以直接跳過。
迭代和迭代器
在理解本文大多數(shù)概念前,有必要知道迭代和迭代器。事實上,迭代大家都知道是什么,可是我不知道(真的,在此之前對這個概念沒有系統(tǒng)了解)。迭代是指反復(fù)執(zhí)行一個過程,每執(zhí)行一次叫做一次迭代。實際上我們經(jīng)常做這種事情,比如:
?php
$mapping = [
'red' => '#FF0000',
'green' => '#00FF00',
'blue' => '#0000FF'
];
foreach ($mapping as $key => $value) {
printf("key: %d - value: %s\n", $key, $value);
}
我們可以看到通過 foreach
對數(shù)組遍歷并迭代輸出其內(nèi)容。在這一環(huán)節(jié)中,我們需要關(guān)注的重點是數(shù)組。雖然我們迭代的過程是 foreach
語句中的代碼塊,但實際上數(shù)組 $mapping
在每一次迭代中發(fā)生了變化,意味著數(shù)組內(nèi)部也存在著一次迭代。如果我們把數(shù)組看做一個對象,foreach 實際上在每一次迭代過程都會調(diào)用該對象的一個方法,讓數(shù)組在自己內(nèi)部進行一次變動(迭代),隨后通過另一個方法取出當(dāng)前數(shù)組對象的鍵和值。這樣一個可通過外部遍歷其內(nèi)部數(shù)據(jù)的對象就是一個迭代器對象,其遵循的統(tǒng)一的訪問接口就是迭代器接口(Iterator
)。
PHP 提供了一個統(tǒng)一的迭代器接口。關(guān)于迭代器 PHP 官方文檔有更為詳細(xì)的描述,建議去了解。
interface Iterator extends Traversable
{
/**
* 獲取當(dāng)前內(nèi)部標(biāo)量指向的元素的數(shù)據(jù)
*/
public mixed current ( void )
/**
* 獲取當(dāng)前標(biāo)量
*/
public scalar key ( void )
/**
* 移動到下一個標(biāo)量
*/
public void next ( void )
/**
* 重置標(biāo)量
*/
public void rewind ( void )
/**
* 檢查當(dāng)前標(biāo)量是否有效
*/
public boolean valid ( void )
}
我們來給出一個實例,去實現(xiàn)一個簡單的迭代器:
class Xrange implements Iterator
{
protected $start;
protected $limit;
protected $step;
protected $i;
public function __construct($start, $limit, $step = 0)
{
$this->start = $start;
$this->limit = $limit;
$this->step = $step;
}
public function rewind()
{
$this->i = $this->start;
}
public function next()
{
$this->i += $this->step;
}
public function current()
{
return $this->i;
}
public function key()
{
return $this->i + 1;
}
public function valid()
{
return $this->i = $this->limit;
}
}
通過 foreach 遍歷來看看這個迭代器的效果:
foreach (new Xrange(0, 10, 2) as $key => $value) {
printf("%d %d\n", $key, $value);
}
輸出:
1 0
3 2
5 4
7 6
9 8
11 10
至此我們看到了一個迭代器的實現(xiàn)。一些人在了解這一特性會很激動的將其應(yīng)用在實際項目中,但有些則疑惑這有什么卵用呢?迭代器只是將一個普通對象變成了一個可被遍歷的對象,這在有些時候,如一個對象 StudentsContact,這個對象是用于處理學(xué)生聯(lián)系方式的,通過 addStudent 方法注冊學(xué)生,通過 getAllStudent 獲取全部注冊的學(xué)生聯(lián)系方式數(shù)組。我們以往遍歷是通過 StudentsContact::getAllStudent() 獲取一個數(shù)組然后遍歷該數(shù)組,但是現(xiàn)在有了迭代器,只要這個類繼承這個接口,就可以直接遍歷該對象獲取學(xué)生數(shù)組,并且可以在獲取之前在類的內(nèi)部就對輸出的數(shù)據(jù)做好處理工作。
當(dāng)然用處遠不止這么點,但在這里就不過多糾結(jié)。有一個在此基礎(chǔ)上更為強大的東西,生成器。
生成器,Generator
雖然迭代器僅需繼承接口即可實現(xiàn),但依舊很麻煩,我們畢竟需要定義一個類并實現(xiàn)該接口所有方法,這十分繁瑣。在一些情景下我們需要更簡潔的辦法。生成器提供了一種更容易的方法來實現(xiàn)簡單的對象迭代,相比較定義類實現(xiàn) Iterator 接口的方式,性能開銷和復(fù)雜性大大降低。
PHP 官方文檔這樣說的:
生成器允許你在 foreach 代碼塊中寫代碼來迭代一組數(shù)據(jù)而不需要在內(nèi)存中創(chuàng)建一個數(shù)組, 那會使你的內(nèi)存達到上限,或者會占據(jù)可觀的處理時間。相反,你可以寫一個生成器函數(shù),就像一個普通的自定義函數(shù)一樣, 和普通函數(shù)只返回一次不同的是, 生成器可以根據(jù)需要 yield 多次,以便生成需要迭代的值。
一個簡單的例子就是使用生成器來重新實現(xiàn) range() 函數(shù)。 標(biāo)準(zhǔn)的 range() 函數(shù)需要在內(nèi)存中生成一個數(shù)組包含每一個在它范圍內(nèi)的值,然后返回該數(shù)組, 結(jié)果就是會產(chǎn)生多個很大的數(shù)組。 比如,調(diào)用 range(0, 1000000) 將導(dǎo)致內(nèi)存占用超過 100 MB。
做為一種替代方法, 我們可以實現(xiàn)一個 xrange() 生成器, 只需要足夠的內(nèi)存來創(chuàng)建 Iterator 對象并在內(nèi)部跟蹤生成器的當(dāng)前狀態(tài),這樣只需要不到1K字節(jié)的內(nèi)存。
官方文檔給了上文對應(yīng)的例子,我們在此簡化了一下:
function xrange($start, $limit, $step = 1) {
for ($i = $start; $i = $limit; $i += $step) {
yield $i + 1 => $i; // 關(guān)鍵字 yield 表明這是一個 generator
}
}
// 我們可以這樣調(diào)用
foreach (xrange(0, 10, 2) as $key => $value) {
printf("%d %d\n", $key, $value);
}
可能你已經(jīng)發(fā)現(xiàn)了,這個例子的輸出和我們前面在說迭代器的時候那個例子結(jié)果一樣。實際上生成器生成的正是一個迭代器對象實例,該迭代器對象繼承了 Iterator 接口,同時也包含了生成器對象自有的接口,具體可以參考 Generator
類的定義。
當(dāng)一個生成器被調(diào)用的時候,它返回一個可以被遍歷的對象.當(dāng)你遍歷這個對象的時候(例如通過一個foreach循環(huán)),PHP 將會在每次需要值的時候調(diào)用生成器函數(shù),并在產(chǎn)生一個值之后保存生成器的狀態(tài),這樣它就可以在需要產(chǎn)生下一個值的時候恢復(fù)調(diào)用狀態(tài)。
一旦不再需要產(chǎn)生更多的值,生成器函數(shù)可以簡單退出,而調(diào)用生成器的代碼還可以繼續(xù)執(zhí)行,就像一個數(shù)組已經(jīng)被遍歷完了。
我們需要注意的關(guān)鍵是 yield
,這是生成器的關(guān)鍵。我們通過上面例子,可以看得出,yield
會將當(dāng)前一個值傳遞給 foreach
,換句話說,foreach
每一次迭代過程都會從 yield 處取一個值,直到整個遍歷過程不再存在 yield 為止的時候,遍歷結(jié)束。
我們也可以發(fā)現(xiàn),yield 和 return 都會返回值,但區(qū)別在于一個 return 是返回既定結(jié)果,一次返回完畢就不再返回新的結(jié)果,而 yield 是不斷產(chǎn)出直到無法產(chǎn)出為止。
實際上存在 yield 的函數(shù)返回值返回的是一個 Generator
對象(這個對象不能手動通過 new 實例化),該對象實現(xiàn)了 Iterator
接口。那么 Generator
自身有什么獨特之處?繼續(xù)看:
yield
字面上解釋,yield 代表著讓位、讓行。正是這個讓行使得通過 yield 實現(xiàn)協(xié)程變得可能。
生成器函數(shù)的核心是 yield 關(guān)鍵字。它最簡單的調(diào)用形式看起來像一個 return 申明,不同之處在于普通 return 會返回值并終止函數(shù)的執(zhí)行,而 yield 會返回一個值給循環(huán)調(diào)用此生成器的代碼并且只是暫停執(zhí)行生成器函數(shù)。
yield 和 return 的區(qū)別,前者是暫停當(dāng)前過程的執(zhí)行并返回值,而后者是中斷當(dāng)前過程并返回值。暫停當(dāng)前過程,意味著將處理權(quán)轉(zhuǎn)交由上一級繼續(xù)進行,直至上一級再次調(diào)用被暫停的過程,該過程則會從上一次暫停的位置繼續(xù)執(zhí)行。這像是什么呢?如果讀者在讀本篇文章之前已經(jīng)在鳥哥的文章中粗略看過,應(yīng)該知道這很像是一個操作系統(tǒng)的進程調(diào)度管理,多個進程在一個 CPU 核心上執(zhí)行,在系統(tǒng)調(diào)度下每一個進程執(zhí)行一段指令就被暫停,切換到下一個進程,這樣看起來就像是同時在執(zhí)行多個任務(wù)。
但僅僅是如此還遠遠不夠,yield 更重要的特性是除了可以返回一個值以外,還能夠接收一個值!
function printer()
{
while (true) {
printf("receive: %s\n", yield);
}
}
$printer = printer();
$printer->send('hello');
$printer->send('world');
上述例子輸出內(nèi)容為:
receive: hello
receive: world
參考 PHP 官方中文文檔:生成器 對象 我們可以得知 Generator 對象除了實現(xiàn) Iterator 接口中的必要方法以外,還有一個 send
方法,這個方法就是向 yield 語句處傳遞一個值,同時從 yied 語句處繼續(xù)執(zhí)行,直至再次遇到 yield 后控制權(quán)回到外部。
我們通過之前也了解了一個問題,yield 可以在其位置中斷并返回一個值,那么能不能同時進行 接收 和 返回 呢?當(dāng)然,這可是實現(xiàn)協(xié)程的根本。我們對上述代碼做出修改:
?php
function printer()
{
$i = 0;
while (true) {
printf("receive: %s\n", (yield ++$i));
}
}
$printer = printer();
printf("%d\n", $printer->current());
$printer->send('hello');
printf("%d\n", $printer->current());
$printer->send('world');
printf("%d\n", $printer->current());
輸出內(nèi)容如下:
1
receive: hello
2
receive: world
3
current
方法是迭代器( Iterator
)接口必要的方法,foreach
語句每一次迭代都會通過其獲取當(dāng)前值,而后調(diào)用迭代器的 next
方法。我們?yōu)榱耸钩绦虿粫o限執(zhí)行,手動調(diào)用 current 方法獲取值。
上述例子已經(jīng)足以表示 yield 在那一個位置作為雙向傳輸?shù)?工具,已具備實現(xiàn)協(xié)程的條件。
協(xié)程
這一部分我不打算長篇大論,本文開頭已經(jīng)給出了鳥哥博客中更為完善的文章,本文的目的是出于補充對 Generator 的細(xì)節(jié)。
我們要知道,對于單核處理器,多任務(wù)的執(zhí)行原理是讓每一個任務(wù)執(zhí)行一段時間,然后中斷、讓另一個任務(wù)執(zhí)行然后在中斷后執(zhí)行下一個,如此反復(fù)。由于其執(zhí)行切換速度很快,讓外部認(rèn)為多個任務(wù)實際上是 “并行” 的。
鳥哥那篇文章這么說道:
多任務(wù)協(xié)作這個術(shù)語中的 “協(xié)作” 很好的說明了如何進行這種切換的:它要求當(dāng)前正在運行的任務(wù)自動把控制傳回給調(diào)度器,這樣就可以運行其他任務(wù)了。這與 “搶占” 多任務(wù)相反, 搶占多任務(wù)是這樣的:調(diào)度器可以中斷運行了一段時間的任務(wù), 不管它喜歡還是不喜歡。協(xié)作多任務(wù)在 Windows 的早期版本 (windows95) 和 Mac OS 中有使用, 不過它們后來都切換到使用搶先多任務(wù)了。理由相當(dāng)明確:如果你依靠程序自動交出控制的話,那么一些惡意的程序?qū)⒑苋菀渍加谜麄€CPU,不與其他任務(wù)共享。
我們結(jié)合之前的例子,可以發(fā)現(xiàn),yield
作為可以讓一段任務(wù)自身中斷,然后回到外部繼續(xù)執(zhí)行。利用這個特性可以實現(xiàn)多任務(wù)調(diào)度的功能,配合 yield
的雙向通訊功能,以實現(xiàn)任務(wù)和調(diào)度器之間進行通信。
這樣的功能對于讀寫和操作 Stream 資源時尤為重要,我們可以極大的提高程序?qū)τ诓l(fā)流資源的處理能力,比如實現(xiàn) tcp server。以上在 《PHP中使用協(xié)同程序?qū)崿F(xiàn)合作多任務(wù)》 有更為詳盡的例子。本文不再贅述。
總結(jié)
PHP 自 5.4 到如今愈發(fā)穩(wěn)定的 PHP 7,可以看到許多的新特性令這門語言愈發(fā)強大和完善,逐漸從純粹的 Web 語言變得有著更為廣泛的適用面,作為一枚 PHPer 的確不應(yīng)當(dāng)止步不前,我們依然有很多的東西需要不斷學(xué)習(xí)和加強。
雖然 “PHP 是世界上最好的語言” 這句話只是個調(diào)侃,但不可否認(rèn) PHP 即使不是最好,但也在努力變好的事實,對吧?
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《php常用函數(shù)與技巧總結(jié)》、《php字符串(string)用法總結(jié)》、《PHP數(shù)組(Array)操作技巧大全》、《PHP數(shù)據(jù)結(jié)構(gòu)與算法教程》及《php程序設(shè)計算法總結(jié)》
希望本文所述對大家PHP程序設(shè)計有所幫助。
您可能感興趣的文章:- PHP Web表單生成器案例分析
- PHP迭代器和生成器用法實例分析
- PHP生成器功能與用法實例分析
- PHP中你可能忽略的性能優(yōu)化利器:生成器
- PHP十六進制顏色隨機生成器功能示例
- PHP新特性詳解之命名空間、性狀與生成器
- php驗證碼生成器
- thinkPHP連接sqlite3數(shù)據(jù)庫的實現(xiàn)方法(附Thinkphp代碼生成器下載)
- PHP5.5迭代生成器用法實例詳解
- PHP 生成器的使用詳解