本文實例講述了PHP迭代器和生成器用法。分享給大家供大家參考,具體如下:
迭代器
迭代器實際是一個實現了Iterator的類,可以用foreach進行遍歷。
例如:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
|
<?php class Sample implements Iterator{ private $curIndex =0; private $items =null; public function __construct( $_items ) { $this ->items = $_items ; } public function current (){ echo "current\n" ; return $this ->items[ $this ->curIndex]; } public function key (){ echo "key\n" ; return $this ->curIndex; } public function next (){ echo "next\n" ; $this ->curIndex++; } public function rewind (){ $this ->curIndex = 0; } public function send ( $value ){ if ( $value == "stop" ){ $this ->curIndex = null; } } public function valid (){ echo "valid\n" ; return isset( $this ->items[ $this ->curIndex]); } } $sample = new Sample([1,2,3]); foreach ( $sample as $k => $v ){ } |
輸出
valid current key next
可以看到foreach 是先調用valid判斷迭代器是否有效,然后再調用current獲取當前值,同時調用next移動key到指向下一個值(輸出key是因為 $k=>$v的緣故)。
生成器
讓我們先看一下官方文檔
生成器提供了一種更容易的方法來實現簡單的對象迭代,相比較定義類實現 Iterator 接口的方式,性能開銷和復雜性大大降低。
生成器允許你在 foreach 代碼塊中寫代碼來迭代一組數據而不需要在內存中創建一個數組, 那會使你的內存達到上限,或者會占據可觀的處理時間。
相反,你可以寫一個生成器函數,就像一個普通的自定義函數一樣, 和普通函數只返回一次不同的是, 生成器可以根據需要 yield 多次,以便生成需要迭代的值。
PHP 將會在每次需要值的時候調用生成器函數,并在產生一個值之后保存生成器的狀態,這樣它就可以在需要產生下一個值的時候恢復調用狀態。
下面是php官方文檔中的示例
1
2
3
4
5
6
7
8
9
10
11
12
|
<?php function gen_one_to_three() { for ( $i = 1; $i <= 3; $i ++) { //注意變量$i的值在不同的yield之間是保持傳遞的。 yield $i ; } } $generator = gen_one_to_three(); foreach ( $generator as $value ) { echo "$value\n" ; } var_dump( $generator ); //實際上是Generator對象 |
如上,若把3修改成10000,對于$generator實際上沒有區別,它只是保存了一個當前值(當然還有相關的內部狀態,這里是為了簡化),并沒有產生10000個數。
從中可以看出生成器的優勢在于減少內存的使用,在需要時才生成對應的值。
查看php文檔,我們可以看到Generator實際也是Iterator的具體實現,yield調用時就是返回的Generator對象。
那么怎么理解迭代器和生成器的關系呢?
其實,生成器是迭代器的實現+yield,產生了生成器對象。
我們也可以自己定義一個類似yield的函數,如下:
1
2
3
4
5
6
7
8
|
function myYeild(){ $args = func_get_args(); return new Sample( $args ); } $generator = myYeild(1,2,3); foreach ( $generator as $value ) { echo "$value\n" ; } |
注意,我們的myYeild,是不能和php內置的yeild那么使用的,因為yeild會保存調用上下文,臨時離開,并沒有return。
這里只是類比一下。
既然yeild可以把普通的對象包裝成generator,那么我們的iterator通過yeild也可以像Generator一樣嗎?
答案有點悲傷,yeild是把傳入的值作為參數生成Generator實例,它并不知道我們的iterator。不過這樣設計也是合理的,
以防我們自己的iterator不靠譜。
實際使用場合
- 數據庫遍歷
可以結合游標,遍歷數據庫時,不需要一次返回所有數據,而是每次取一行。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
|
class AllUser implements \Iterator { protected $index = 0; protected $data = []; public function __construct() { $link = mysqli_connect( '192.168.0.91' , 'root' , '123' , 'xxx' ); $rec = mysqli_query( $link , 'select id from doc_admin' ); $this ->data = mysqli_fetch_all( $rec , MYSQLI_ASSOC); } //1 重置迭代器 public function rewind () { $this ->index = 0; } //2 驗證迭代器是否有數據 public function valid() { return $this ->index < count ( $this ->data); } //3 獲取當前內容 public function current() { $id = $this ->data[ $this ->index]; return User::find( $id ); } //4 移動key到下一個 public function next() { return $this ->index++; } //5 迭代器位置key public function key() { return $this ->index; } } //實現迭代遍歷用戶表 $users = new AllUser(); //可實時修改 foreach ( $users as $user ){ $user ->add_time = time(); $user ->save(); } |
-
文件遍歷
一次讀取一行 - 實現Iterator接口,讓普通類可以使用foreach遍歷。
- 協程,參見鳥哥則這篇文章。
注意:可以在生成器的函數前加"&",可以使用引用。在函數里直接return會終止生成器。
希望本文所述對大家PHP程序設計有所幫助。
原文鏈接:https://www.cnblogs.com/xdao/p/php_iterator_generator.html