PHP闭包(Closure)使用详解


Posted in PHP onMay 02, 2013

不知不觉发现PHP已经出到了5.5版本,而自己一直在用PHP5.2,让我看起来像深山出来的小伙子一样,又土又落后。在我习惯在javascript中使用闭包之后,忽然间对PHP的闭包打起了兴趣。

于是乎在网上下了个WAMP集成开发环境,是PHP5.3版本的(PHP5.3开始引入了闭包的特性),不得不说WAMP安装使用真的很方便。简单配置了一下,开始动手。
 

匿名函数
提到闭包就不得不想起匿名函数,也叫闭包函数(closures),貌似PHP闭包实现主要就是靠它。声明一个匿名函数是这样:

$func = function() {}; //带结束符

可以看到,匿名函数因为没有名字,如果要使用它,需要将其返回给一个变量。匿名函数也像普通函数一样可以声明参数,调用方法也相同:
$func = function( $param ) {
    echo $param;
};
$func( 'some string' );
//输出:
//some string

顺便提一下,PHP在引入闭包之前,也有一个可以创建匿名函数的函数:create function,但是代码逻辑只能写成字符串,这样看起来很晦涩并且不好维护,所以很少有人用。
 

实现闭包
将匿名函数在普通函数中当做参数传入,也可以被返回。这就实现了一个简单的闭包。

下边有三个例子

//例一
//在函数里定义一个匿名函数,并且调用它
function printStr() {
    $func = function( $str ) {
        echo $str;
    };
    $func( 'some string' );
}
printStr();
 
//例二
//在函数中把匿名函数返回,并且调用它
function getPrintStrFunc() {
    $func = function( $str ) {
        echo $str;
    };
    return $func;
}
$printStrFunc = getPrintStrFunc();
$printStrFunc( 'some string' );
 

//例三
//把匿名函数当做参数传递,并且调用它
function callFunc( $func ) {
    $func( 'some string' );
}
$printStrFunc = function( $str ) {
    echo $str;
};
callFunc( $printStrFunc );
//也可以直接将匿名函数进行传递。如果你了解js,这种写法可能会很熟悉
callFunc( function( $str ) {
    echo $str;
} );

连接闭包和外界变量的关键字:USE
闭包可以保存所在代码块上下文的一些变量和值。PHP在默认情况下,匿名函数不能调用所在代码块的上下文变量,而需要通过使用use关键字。

换一个例子看看:

function getMoney() {
    $rmb = 1;
    $dollar = 6;
    $func = function() use ( $rmb ) {
        echo $rmb;
        echo $dollar;
    };
    $func();
}
getMoney();
//输出:
//1
//报错,找不到dorllar变量

可以看到,dollar没有在use关键字中声明,在这个匿名函数里也就不能获取到它,所以开发中要注意这个问题。
 

有人可能会想到,是否可以在匿名函数中改变上下文的变量,但我发现是不可以的:

function getMoney() {
    $rmb = 1;
    $func = function() use ( $rmb ) {
        echo $rmb;
        //把$rmb的值加1
        $rmb++;
    };
    $func();
    echo $rmb;
}
getMoney();
//输出:
//1
//1

啊,原来use所引用的也只不过是变量的一个副本而已。但是我想要完全引用变量,而不是复制。

要达到这种效果,其实在变量前加一个 & 符号就可以了:

function getMoney() {
    $rmb = 1;
    $func = function() use ( &$rmb ) {
        echo $rmb;
        //把$rmb的值加1
        $rmb++;
    };
    $func();
    echo $rmb;
}
getMoney();
//输出:
//1
//2

好,这样匿名函数就可以引用上下文的变量了。如果将匿名函数返回给外界,匿名函数会保存use所引用的变量,而外界则不能得到这些变量,这样形成‘闭包'这个概念可能会更清晰一些。

根据描述改变一下上面的例子:

function getMoneyFunc() {
    $rmb = 1;
    $func = function() use ( &$rmb ) {
        echo $rmb;
        //把$rmb的值加1
        $rmb++;
    };
    return $func;
}
$getMoney = getMoneyFunc();
$getMoney();
$getMoney();
$getMoney();
//输出:
//1
//2
//3

总结
PHP闭包的特性并没有太大惊喜,其实用CLASS就可以实现类似甚至强大得多的功能,更不能和js的闭包相提并论,只能期待PHP以后对闭包支持的改进。不过匿名函数还是挺有用的,比如在使用preg_replace_callback等之类的函数可以不用在外部声明回调函数了。
PHP 相关文章推荐
PHP中的超全局变量
Oct 09 PHP
Adodb的十个实例(清晰版)
Dec 31 PHP
php 文件上传系统手记
Oct 26 PHP
PHP数据流应用的一个简单实例
Sep 14 PHP
PHP 自定义错误处理函数trigger_error()
Mar 26 PHP
php中adodbzip类实例
Dec 08 PHP
php通过rmdir删除目录的简单用法
Mar 18 PHP
php无限分类使用concat如何实现
Nov 05 PHP
PHP编程实现的TCP服务端和客户端功能示例
Apr 13 PHP
PHP实现负载均衡下的session共用功能
Apr 17 PHP
Laravel修改验证提示信息为中文的示例
Oct 23 PHP
如何判断微信付款码和支付宝付款码
Apr 01 PHP
PHP5中Cookie与 Session使用详解
Apr 30 #PHP
PHP容易忘记的知识点分享
Apr 30 #PHP
基于curl数据采集之正则处理函数get_matches的使用
Apr 28 #PHP
基于curl数据采集之单页面并行采集函数get_htmls的使用
Apr 28 #PHP
基于curl数据采集之单页面采集函数get_html的使用
Apr 28 #PHP
基于php上传图片重命名的6种解决方法的详细介绍
Apr 28 #PHP
PHP基础学习之流程控制的实现分析
Apr 28 #PHP
You might like
PHP 最大运行时间 max_execution_time修改方法
2010/03/08 PHP
PHP支持多种格式图片上传(支持jpg、png、gif)
2011/11/03 PHP
PHP跳转页面的几种实现方法详解
2013/06/08 PHP
php使用curl模拟登录后采集页面的例子
2013/11/04 PHP
Yii实现显示静态页的方法
2016/04/25 PHP
php lcg_value与mt_rand生成0~1随机小数的效果对比分析
2017/04/05 PHP
PC端微信扫码支付成功之后自动跳转php版代码
2017/07/07 PHP
js 对联广告、漂浮广告封装类(IE,FF,Opera,Safari,Chrome
2009/11/26 Javascript
浅谈Javascript数组(推荐)
2016/05/17 Javascript
Javascript之Number对象介绍
2016/06/07 Javascript
JavaScript实现鼠标点击导航栏变色特效
2017/02/08 Javascript
NodeJS创建最简单的HTTP服务器
2017/05/15 NodeJs
js实现图片旋转 js滚动鼠标中间对图片放大缩小
2017/07/05 Javascript
IScroll5实现下拉刷新上拉加载的功能实例
2017/08/11 Javascript
浅谈vue首屏加载优化
2018/06/28 Javascript
用图片替换checkbox原始样式并实现同样的功能
2018/11/15 Javascript
利用vue3+ts实现管理后台(增删改查)
2020/10/30 Javascript
使用Python实现下载网易云音乐的高清MV
2015/03/16 Python
python学习入门细节知识点
2018/03/29 Python
python+pandas生成指定日期和重采样的方法
2018/04/11 Python
在PyTorch中Tensor的查找和筛选例子
2019/08/18 Python
python lambda表达式在sort函数中的使用详解
2019/08/28 Python
python实现按关键字筛选日志文件
2019/12/24 Python
TensorFlow2.0矩阵与向量的加减乘实例
2020/02/07 Python
python判断变量是否为int、字符串、列表、元组、字典的方法详解
2020/02/13 Python
matplotlib对象拾取事件处理的实现
2021/01/14 Python
CSS3的一个简单导航栏实现
2015/08/03 HTML / CSS
boostrap modal 闪现问题的解决方法
2020/09/01 HTML / CSS
美国床垫连锁店:Mattress Firm
2021/02/13 全球购物
Pharmacy Online中文直邮网站:澳洲大型药房
2020/06/27 全球购物
C#中类(class)与结构(struct)的异同
2013/11/03 面试题
学生个人自我鉴定范文
2014/03/28 职场文书
股东协议书范本2016
2016/03/21 职场文书
再见,2019我们不负使命;你好,2020我们砥砺前行
2020/01/03 职场文书
使用vue-element-admin框架从后端动态获取菜单功能的实现
2021/04/29 Vue.js
Golang实现可重入锁的示例代码
2022/05/25 Golang