官方解釋yield
yield生成器
是php5.5
之後出現的,官方文檔這樣解釋:yield
提供了一種更容易的方法來實現簡單的迭代對象,相比較定義類實現Iterator
介面的方式,效能開銷和複雜度大大降低。
生成器的核心是一個yield
關鍵字,一個生成器函數看起來像一個普通的函數,不同的是:普通函數回傳一個值,而一個生成器可以yield
產生許多它所需的值。生成器函數被呼叫時,傳回的是一個可以被遍歷的物件。
yield
和return
有點類似,不過不同的是,return
會傳回值並且終止程式碼的執行,而yield
會傳回一個值給循環呼叫此生成器的程式碼並且只是暫停執行生成器函數。
這裡順便跟大家介紹一下php版本的非緩衝查詢
意思就是把資料一行行讀取到php運行內存,並非一次讀取到php運行內存,眾所周知,php有很多內置函數,可以幫助我們對數據進行加工操作,因為數據都在內存裡面,所以能操作,但是php的運行內存是有極限,默認128M。
注意:因為非緩衝查詢是會長時間連接資料庫的,有可能會造成慢查詢、鎖定表之類的情況,比較耗mysql資源
相對非緩衝查詢就是緩衝查詢:
如果用快取查詢,php記憶體就會直接爆了,出現記憶體不足的情況。好了,這裡主要是為了突出yield
yield效能
產生器會對PHP應用的效能有非常大的影響
PHP程式碼運行時節省大量的記憶體
比較適合計算大量的資料
yield運用
生成器允許你在foreach 程式碼區塊中寫程式碼來迭代一組資料而不需要在記憶體中創建一個數組,那會使你的記憶體達到上限,或者會佔據可觀的處理時間。相反,你可以寫一個生成器函數,就像一個普通的自定義函數一樣, 和普通函數只返回一次不同的是, 生成器可以根據需要 yield 多次,以便生成需要迭代的值。
範例解說
我這裡只是建立了一個陣列來給大家示範,平常你在操作資料庫輸出資料也是一樣的。轉為array
//仓库库存扣除测试 public function cangku_stock() { //set_time_limit(0); //表示永久运行,这里我是测试array的时候用到的 $order_info = $this->read_temp_api_order_info(10); //这里我就测试了10条数据,效果是看不出来的 foreach($order_info as $temp_api_order_info){ dd($temp_api_order_info); //打印出来看看数据 //处理数据 $api_ware_id = $this->o->getCangkuApiUrl() .'ware/program/addOutWare'; $out_wares = api_request($api_ware_id, $temp_api_order_info); $temp_out_wares = json_decode($out_wares, true); if ($temp_out_wares['code'] != 1) { $msg = (isset($temp_out_wares['msg']) && $temp_out_wares['msg']) ? $temp_out_wares['msg'] : var_export($out_wares, true); throw new Exception($msg); } } //dd("批量更新成功".date('Y-m-d H:i:s'));
可以看到我們呼叫$order_info = $this->read_temp_api_order_info(10);傳回了一個Generator 對象,這個物件可以使用foreach 迭代,每次迭代,PHP 會要求Generator 實例計算並提供下一個要迭代的值。
生成器的優雅體現在每次產出一個值之後,生成器的內部狀態都會停頓;當產生器請求下一個值時,內部狀態又會恢復。生成器內部的狀態會一直在停頓和復原之間切換,直到抵達函數定義體的末端或遇到空的 return 語句為止。
效果如下:
這裡測試大量數據,直接更改$this->read_temp_api_order_info(10);就
好,如果是計算資料表數量,那你就要改改這個方法了。自己試著更改一下吧。
這裡我主要跟大家講解yield的用法,如果要看插入資料表花了多久時間,自己可以在資料表增加插入時間的字段,再看看第一條資料插入與最後一條資料插入的時候的比較的。
更多相關php知識,請造訪php教學!
以上是PHP生成器yield處理大量資料業務(程式碼範例)的詳細內容。更多資訊請關注PHP中文網其他相關文章!