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
php&java(二)
Oct 09 PHP
两个强悍的php 图像处理类1
Jun 15 PHP
PHP 事件机制(2)
Mar 23 PHP
PHP setcookie设置Cookie用法(及设置无效的问题)
Jul 13 PHP
将时间以距今多久的形式表示,PHP,js双版本
Sep 25 PHP
封装ThinkPHP的一个文件上传方法实例
Oct 31 PHP
腾讯微博提示missing parameter errorcode 102 错误的解决方法
Dec 22 PHP
php HTML无刷新提交表单
Apr 05 PHP
Phpstorm+Xdebug断点调试PHP的方法
May 14 PHP
Laravel5.7 Eloquent ORM快速入门详解
Apr 12 PHP
TP5框架安全机制实例分析
Apr 05 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添加MySQL数据记录代码
2008/06/07 PHP
Php Image Resize图片大小调整的函数代码
2011/01/17 PHP
PHP文章按日期(月日)SQL归档语句
2012/11/29 PHP
浅谈PHP的$_SERVER[SERVER_NAME]
2017/02/04 PHP
简单谈谈PHP面向对象之标识对象
2017/06/27 PHP
Laravel5.5+ 使用API Resources快速输出自定义JSON方法详解
2020/04/06 PHP
JQuery实现自定义对话框的代码
2008/06/15 Javascript
js 单引号 传递方法
2009/06/22 Javascript
JavaScript实现页面实时显示当前时间的简单实例
2013/07/20 Javascript
对于Form表单reset方法的新认识
2014/03/05 Javascript
js改变透明度实现轮播图的算法
2020/08/24 Javascript
基于jQuery ligerUI实现分页样式
2016/09/18 Javascript
JavaScript常用正则函数用法示例
2017/01/23 Javascript
ES6新特性七:数组的扩充详解
2017/04/21 Javascript
JavaScript中使用Async实现异步控制
2017/08/15 Javascript
详解Vue单元测试Karma+Mocha学习笔记
2018/01/31 Javascript
详解vue后台系统登录态管理
2019/04/02 Javascript
关于Vue源码vm.$watch()内部原理详解
2019/04/26 Javascript
微信小程序获取用户信息并保存登录状态详解
2019/05/10 Javascript
jsonp跨域获取百度联想词的方法分析
2019/05/13 Javascript
vuex管理状态 刷新页面保持不被清空的解决方案
2019/11/11 Javascript
Node.js中console.log()输出彩色字体的方法示例
2019/12/01 Javascript
vuex的使用步骤
2021/01/06 Vue.js
[52:05]EG vs OG 2019国际邀请赛小组赛 BO2 第二场 8.16
2019/08/18 DOTA
Python连接phoenix的方法示例
2017/09/29 Python
Django中使用CORS实现跨域请求过程解析
2019/08/05 Python
Python如何基于selenium实现自动登录博客园
2019/12/16 Python
Python文本处理简单易懂方法解析
2019/12/19 Python
Python如何使用字符打印照片
2020/01/03 Python
python调用百度API实现人脸识别
2020/11/17 Python
python中xlrd模块的使用详解
2021/02/01 Python
Foot Locker英国官网:美国知名运动产品零售商
2019/02/21 全球购物
大学生会计职业生涯规划范文
2014/02/28 职场文书
《叶问2》观后感
2015/06/15 职场文书
2015教师个人师德工作总结
2015/10/23 职场文书
Pytorch反向传播中的细节-计算梯度时的默认累加操作
2021/06/05 Python