成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

談PHP閉包特性在實際應用中的問題

開發 后端 前端
很多語言的都提供了非常優雅和漂亮的操作數組的方法,文章將以PHP 5.3為例以及其他語言提供的閉包功能,用于展示如何“客觀的”操作迭代數組。

PHP 5.3版本跟隨了很多新特性,其中比較惹眼的特性之一就是支持了閉包。文章將使用PHP 5.3 以及其他語言提供的閉包功能,用于展示如何“客觀的”操作迭代數組。在開始之前先說明下,本例子僅僅是闡明觀點,并沒有考慮性能等其他方面的因素。

51CTO推薦專題: PHP開發基礎入門

“貨比三家”

用個簡單的例子開始,有下面個數組:

  1. $nums = array(10, 20, 30, 40); 
  2.  

需要找出數組中大于15的項。那么,不考慮閉包的情況下,我們或許會這樣寫:

  1. $res = array();foreach ($nums as $n)   
  2. {      
  3. if ($n > 15)   
  4. {          
  5. $res[] = $n;      
  6. }} 

如果語言本身有閉包支持的,那么或許會這樣寫(Groovy 語言)

  1. def res = nums.findAll { it > 15 } 
  2.  

或者使用Scala語言:

  1. val res = nums filter (_ > 15) 
  2.  

譯注:Javascript 1.6 的話會是如下:

  1. var res = nums.filter(function(c){return c > 15}); 
  2.  

因為循環操作已被抽象起來,所以可以看到 Groovy 、Scala (以及 Javascript) 都很漂亮得用一行就可以搞定。當然,如果使用 PHP5.3 的閉包,也可以做到

  1. $res = array_filter($nums, function($v) { return $v > 15; }); 
  2.  

PHP在這方面使用了比Scala更多的字符,但對比先前的例子,它更簡短并且能更好得閱讀。

順便說下,上面的PHP代碼實際上是使用了Lambda解析式,并不是個真正的閉包,這個并不是我們目前關注的重點。目前看來感覺都還不錯,那么我們再的題目增加點難度:找到所有大于15的項, 然后乘以2再加上作用域中的的某個變量值以后再返回。

Groovy的實現:

  1. def x = 1def res = nums .findAll { it > 15 } .collect { it * 2 + x } 
  2.  

Scala的實現:

  1. val x = 1val res = nums filter (_ > 15) map (_ * 2 + x)
  2.  

Javascript 的實現:

  1. var i = 1;var res = nums.filter(function(c)
  2. {return c > 15}).map(function(c){return c * 2 + i}); 

PHP的實現:

  1. $x = 1;$res = array_map(      
  2. function($v) use ($x) { return $v * 2 + $x;   
  3. },    array_filter(        $nums,        function($v) { return $v > 15; }  
  4.  
  5. )); 

光從代碼量方面,現在看起來PHP與其他語言有出入了。先拋開代碼字面上本身的審美不談,上面的PHP代碼還有個額外的問題。例如,如果需要使用數組的鍵而非值作比較,怎么辦?是的,上面的代碼就辦不到了。同時,從語法角度上說,上面的代碼非常難以閱讀。返璞歸真,這時還是得返回老土的思路去解決問題:

  1. $x = 1;$res = array();foreach ($nums as $n)   
  2. {    if ($n > 15) {        $res[] = $n * 2 + $x;    }} 

這樣看起來又很清楚了。但這個時候你或許又會迷惑了:“那還瞎折騰啥,這不就是個數組操作嗎?”。是的,好戲還在后頭。這個時候該讓 PHP 的某些高級特性出場,來搞定這看似有自殘傾向 的“無聊問題”。

#p#

ArrayObject – 對數組的封裝

PHP有個稱作SPL的標準庫,其中包含了個叫做ArrayObject的類,它能提供“像數組一 樣操作類”的功能,例如

  1. $res = new ArrayObject(array(10, 20, 30, 40));foreach ($res as $v) {  
  2.    echo "$vn";} 

ArrayObject是個內置的類,所以你可以像其他類類操作一樣封裝它。

Arr - 包上糖衣

既然我們已經有了ArrayObject以及閉包這些特性,我們就可以開始嘗試封裝它:

  1. class Arr extends ArrayObject  
  2. {      
  3. static function make($array)      
  4. {          
  5. return new self($array);      
  6. }    function map($func)      
  7. {          
  8. $res = new self();          
  9. foreach ($this as $k => $v)   
  10. {              
  11. $res[$k] = $func($k, $v);          
  12. }          
  13. return $res;      
  14. }      
  15. function filter($func)      
  16. {          
  17. $res = new self();         
  18.  foreach ($this as $k => $v)   
  19. {              
  20. if ($func($k, $v))   
  21. {                  
  22. $res[$k] = $v;             
  23.  }          
  24.  
  25. }          
  26.  
  27. return $res;      
  28. }  

好了,萬事俱備。下面重寫的PHP代碼就可以解決上面提到的問題,并且看起來語法上“差 不多”了:

  1. $res = Arr::make($nums)      
  2. ->filter(function($k, $v) { return $v > 15; })      
  3. ->map(function($k, $v) { return $v * 2; }); 

上面的代碼與傳統方式有何不同呢?首先,它們可以遞歸并形成作用鏈式的調用,因此可以 添加更多的類似操作。同時,可以通過回調的兩個參數分別操作數組的鍵以及值其項 - $k 對應鍵以及 $v 對應值 。這使得我們可以在閉包中使用鍵值,這在傳統的PHP函數 array_fliter中是無法實現的。

另外個帶來的額外好處就是更加一致API調用。使用傳統的 PHP 函數操作,它們有可能第一個參數是個閉包,或者是個數組,抑或是多個數組…總之誰知道呢?這里是 Arr 類的完整源代碼,還包含了其他有用的函數(類似 reduce 以及 walk),其實它 們的實現其實方式和代碼類似。

#p#

博弈

這個問題其實很難回答 - 這需要根據代碼的上下文以及程序員自身等眾多因素決定。其實 ,當我第一眼看見PHP的閉包實現時,我感覺似乎回到了那很久以前的Java時期,當時 我在開始使用匿名內置類(anonymous inner classes)來實現閉包。當然,這雖然可以做到, 但看起來實在是些畫蛇添足。PHP 閉包本身是沒錯,只是它的實現以及語法讓我感到非常的困惑。

其他具有閉包特性的語言,它們可以非常方便的調用閉包并同時具有優雅的語法。在上面的例子中,在 Scala中使用傳統的循環也可以工作,但你會這樣寫嗎?而從另個方面,那么有人 說上面這個題目使用 PHP 的閉包也可以實現,但一般情況下你會這樣寫嗎?

可以確定,PHP 閉包在些情況下可以成為銳利的軍刀(例如延時執行以及資源調用方面), 但在傳統的迭代以及數組操作面前就顯得有些為難。不要氣餒不管怎么樣, 返璞歸真編寫具有兼容性的、清爽的代碼以及 API 是最重要的。

結束語

像所有后來加上的語法特性一樣(記得當年Java的Generics特性不?以及前幾年的PHP OOP特性),它們都需要時間磨合以及最終穩定下來。隨著PHP5.3甚至將來的PHP 6逐漸普及,越來越多的技巧和特性相信在不遠的將來逐漸的被聰明的程序員挖掘出來。回到最初文章開頭那個題目,對比:

  1. $res = Arr::make($nums)      
  2. ->filter(function($k, $v)   
  3. { return $v > 15; })      
  4. ->map(function($k, $v)   
  5. { return $v * 2; });以及  
  6. val res = nums filter (_ > 15) map (_ * 2) 

兩者之間的區別。歸根結底它們僅是語法而已,本質上都是殊途同歸解決了同個問題。程序語言的應用特性不同,自然孰優孰劣也就無從比較。最后,這里有此篇文章的代碼示例, 相信可以找到更多如何使用PHP進行函數式迭代(當然不僅僅是這些)的心得。

文章轉自李魁的博客,

原文地址:http://www.cahlk.com.cn/2010/984.shtml

【編輯推薦】

  1. 那些未曾了解的PHP函數和功能
  2. 關于PHP你可能不知道的10件事
  3. PHP開發人員容易忽略的幾點精華
  4. Nginx + PHP-FPM + APC=絕妙的組合 
責任編輯:王曉東 來源: 李魁的博客
相關推薦

2009-10-26 09:41:26

PHP5.3閉包特性

2009-12-02 19:51:54

PHP Switch語

2009-07-24 17:30:37

Javascript閉

2010-05-06 16:02:42

Oracle SQL

2009-12-01 14:46:16

PHP mb_subs

2013-05-02 09:44:57

PHP閉包

2010-03-18 13:49:40

ATM交換機

2011-03-02 12:33:00

JavaScript

2009-11-27 10:02:54

PHP mb_strw

2009-11-26 15:35:43

PHP函數eregi(

2019-01-16 07:39:14

2009-05-13 14:15:09

PHP 5.3閉包匿名函數

2016-09-14 09:20:05

JavaScript閉包Web

2021-08-28 10:06:29

VueJavascript應用

2009-11-03 11:03:00

CDN接入技術

2024-02-27 16:27:42

物聯網IOT智能連接

2009-11-23 14:17:50

PHP 5.3閉包語法

2010-01-06 15:21:00

軟交換技術

2020-02-12 16:58:15

JavaScript前端技術

2009-11-26 09:06:35

PHP遞歸數組
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 狠狠操婷婷 | 色婷婷影院 | 午夜激情影院 | 久久精品国产一区二区电影 | 亚洲三区在线观看 | 一区二区三区在线看 | 一级片av| 国产成人精品午夜 | 色婷婷精品国产一区二区三区 | 91精品国产乱码久久久久久久 | 亚洲a级 | 欧美一区二区在线 | 黄免费观看视频 | 久久久tv | 毛片一区二区 | 怡红院成人在线视频 | 亚洲国产视频一区二区 | 日韩精品在线观看一区二区 | 黄色网址av | 欧美日韩国产精品一区 | 成人激情视频免费观看 | 久久国内 | 中文字幕一区在线观看视频 | 午夜国产一级片 | 国产日韩欧美中文 | 久久精品久久久久久 | 午夜精品视频在线观看 | 国产精品一区二区在线观看 | 亚洲国产情侣 | 成人在线a| 亚洲精品一区二区三区 | 激情五月综合 | www.国产一区| 国产 日韩 欧美 在线 | 在线只有精品 | 日韩一区不卡 | 欧美激情啪啪 | 国产成人99久久亚洲综合精品 | 色一情一乱一伦一区二区三区 | 久久久精品日本 | www.97国产|