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

PHP5.3閉包特性及應用詳解

開發 后端
隨著PHP5.3和即將到來的PHP6的普及,越來越多的特性和技巧將被程序員們熟知和使用,本文向您介紹的是PHP5.3閉包特性,熟練使用,可以使PHP成為Web開發的銳利軍刀。

今年7月發布的PHP 5.3 版本帶來了很多新的特性,, 其中比較惹眼的特性之一就是支持了閉包;51CTO曾經發布的《PHP 5.3閉包語法初探》一文也受到了PHP開發者的歡迎。

那么以后,我們也可以和那幫寫 Ruby、Javascript 等等“高科技語言”的家伙們一樣,寫出非??岬拇a嗎?呃,其實大部分情況下是可以的,而有些方面還是令人非常的困擾,下面慢慢道來。

很多語言的都提供了非常優雅和漂亮的操作數組的方法。在下面的例子中,會使用PHP5.3閉包特性以及其他語言提供的閉包功能,用于展示如何“客觀的”操作迭代數組。

譯注:原文作者比較火星,我不了解 Groovy 以及 Scala語言,所以這里我加上 Javascript 的實現。

在開始之前先說明下,本例子僅僅是闡明觀點,并沒有考慮性能等其他方面的因素。

“貨比三家”

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

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

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

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

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

def res = nums.findAll { it > 15 }或者使用 Scala 語言

val res = nums filter (_ > 15)譯注:Javascript 1.6 的話會是如下

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

因為循環操作已被抽象起來,所以可以看到 Groovy 、Scala (以及 Javascript) 都很漂亮得用一行就可以搞定。

當然,如果使用 PHP5.3 的閉包,也可以做到

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

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

順便說下,上面的 PHP 代碼實際上是使用了 Lambda 解析式,并不是個真正的閉包,這個 并不是我們目前關注的重點。詳細闡述 PHP 閉包以及 Lambda 解析式的資料,可以參考這里。

目前看來感覺都還不錯,那么我們再的題目增加點難度:找到所有大于 15 的項, 然后乘以 2 再加上作用域中的的某個變量值以后再返回。

Groovy 的實現:

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

Scala 的實現:

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

PHP的實現:

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

光從代碼量方面,現在看起來 PHP 與其他語言有出入了。先拋開代碼字面上本身 的審美不談,上面的 PHP 代碼還有個額外的問題。

例如,如果需要使用數組的鍵而非值作比較,怎么辦?是的,上面的代碼就辦不到了。同時,從語法角度上說,上面的代碼非常難以閱讀。

返璞歸真,這時還是得返回老土的思路去解決問題:

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

這樣看起來又很清楚了。但這個時候你或許又會迷惑了:“那還瞎折騰啥,這不就是個數組操作嗎?”。

是的,好戲還在后頭。這個時候該讓 PHP 的某些高級特性出場,來搞定這看似有自殘傾向 的“無聊問題”。

ArrayObject – 對數組的封裝

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

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

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

Arr - 包上糖衣

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

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

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

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

上面的代碼與傳統方式有何不同呢?首先,它們可以遞歸并形成作用鏈式的調用,因此可以 添加更多的類似操作。

同時,可以通過回調的兩個參數分別操作數組的鍵以及值其項 - $k 對應鍵以及 $v 對應值 。這使得我們可以在閉包中使用鍵值,這在傳統的 PHP 函數 array_fliter 中是無法實現的。

另外個帶來的額外好處就是更加一致 API 調用。使用傳統的 PHP 函數操作,它們有可能第一個參數是個閉包,或者是個數組,抑或是多個數組…總之誰知道呢?

這里是 Arr 類的完整源代碼,還包含了其他有用的函數(類似 reduce 以及 walk),其實它 們的實現其實方式和代碼類似。

博弈

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

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

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

結束語

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

回到最初文章開頭那個題目,對比

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

以及

val res = nums filter (_ > 15) map (_ * 2)

兩者之間的區別。歸根結底它們僅是語法而已,本質上都是殊途同歸解決了同個問題。程序 語言的應用特性不同,自然孰優孰劣也就無從比較。

 

【編輯推薦】

  1. PHP 5.3閉包語法初探
  2. PHP 5.3中的命名空間:你用過了么?
  3. PHP 5.3:遠大于開發者預期的重大版本
  4. Windows 7擁抱PHP 三大特征打造高效IT
  5. PHP include()和require()方法的區別
責任編輯:佚名 來源: PHP研究室
相關推薦

2009-05-13 14:15:09

PHP 5.3閉包匿名函數

2009-11-23 14:17:50

PHP 5.3閉包語法

2009-03-23 10:53:00

PHP5.3新特性PHP5.3PHP5

2010-06-17 10:22:47

PHP

2011-07-27 16:55:12

Objective-c 閉包

2013-05-30 01:01:31

2013-05-02 09:44:57

PHP閉包

2009-07-24 17:30:37

Javascript閉

2011-02-15 09:23:55

PHPPhpStorm

2009-07-17 09:31:06

PHP 5.3命名空間

2020-02-12 16:58:15

JavaScript前端技術

2011-03-02 12:33:00

JavaScript

2016-09-14 09:20:05

JavaScript閉包Web

2009-08-25 10:58:41

PHP命名空間

2022-10-10 09:00:29

SendableSwift

2024-07-10 08:31:59

C#特性代碼

2024-10-21 07:05:14

C#特性語言

2024-01-22 09:51:32

Swift閉包表達式尾隨閉包

2010-09-10 16:21:58

JavaScript

2010-06-29 08:39:25

Python閉包
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 在线观看的av| 欧美日批 | 国产一区2区 | 国产精品久久久久久二区 | 农夫在线精品视频免费观看 | 久久免费视频网 | 综合网视频 | 免费h在线| 日韩在线欧美 | 久久99精品久久久久蜜桃tv | 精品国产伦一区二区三区观看说明 | 欧美日韩在线国产 | 国产毛片在线看 | 国产精彩视频 | 一级毛片大全免费播放 | 精品在线观看入口 | 粉嫩av在线 | 久操伊人 | 日本黄色的视频 | 一区二区三区四区免费在线观看 | jvid精品资源在线观看 | 亚洲国产成人精品久久久国产成人一区 | 国内自拍偷拍 | 精品国产伦一区二区三区观看说明 | 一级a性色生活片久久毛片波多野 | 欧洲精品在线观看 | 国产一区二区三区色淫影院 | 91精品国产91久久久久久最新 | 国家一级黄色片 | 国产精品久久亚洲7777 | 国内精品视频一区二区三区 | 日韩精品一区二区三区中文在线 | 国产在线一| 日韩av一区二区在线观看 | 中文字幕黄色大片 | 国产激情视频网 | 欧美国产视频一区二区 | 国产a级毛毛片 | 成人欧美日韩一区二区三区 | 99精品视频在线 | 亚洲九九精品 |