php 闭包
创始人
2024-02-11 22:55:44
0

一、概念说明

通常定义php函数时,都会指定一个函数名,这样的函数可以称为具名函数,但实际上PHP也支持定义没有函数名的函数,这类函数被称为闭包,也叫匿名函数,其本质是 Closure 类对象,类摘要如下

// 类用final修饰,防止定义子类
final class Closure {// 构造函数被定义为私有,防止匿名函数被实例化private __construct ( void )public static bind ( Closure $closure , object $newthis [, mixed $newscope = "static" ] ) : Closurepublic bindTo ( object $newthis [, mixed $newscope = "static" ] ) : Closurepublic call ( object $newthis [, mixed $... ] ) : mixedpublic static fromCallable ( callable $callable ) : Closurepublic __invoke( ...$values): mixed
}

二、函数说明

2.1 __invoke

当以调用函数的方式调用一个对象时,__invoke() 方法会被自动调用,直接调用__invoke 也是一样会执行编写的函数体,下面是示例

function callInvoke(){$func = function ($name,$age){echo __FUNCTION__," name=$name,age=$age\n";};$func->__invoke("kate",100);
}callInvoke();

输出结果

{closure} name=kate,age=100

所以,编写的匿名函数函数体可以被认为是 __invoke 函数的函数体。

2.2 bindTo

返回一个新的闭包,新的闭包绑定了指定的 this对象和类作用域。也就是匿名函数中this 对象和类作用域。也就是匿名函数中this对象和类作用域。也就是匿名函数中this表示的是被绑定的对象,可以直接使用其成员变量和成员函数,反过来,相当于指定的对象临时增加了一个新的方法,只不过只能用普通的函数的调用方式来调用,某种程度上扩展了对象的功能。

// 定义商品类
class Good {private $price;public function __construct(float $price){$this->price = $price;}
}// 定义一个匿名函数,计算商品的促销价
$addDiscount = function(float $discount = 0.8){return $this->price * $discount;
}$good = new Good(100);// 将匿名函数绑定到 $good 实例,同时指定作用域为 Good
$count = $addDiscount->bindTo($good, Good::class); 
$count(); // 80// 将匿名函数绑定到 $good 实例,但是不指定作用域,将无法访问 $good 的私有属性
$count = $addDiscount->bindTo($good); 
$count(); // 报错

2.3 bind

该函数是 bindTo 方法的静态版本,有两种用法:
用法一:实现与 bindTo 方法同样的效果

$count = \Closure::bind($addDiscount, $good, Good::class); 

用法二:将匿名函数与类(而不是对象)绑定,此时第二个参数需要设置为 null

// 商品库存为 10
class Good {static $num = 10;
}// 每次销售后返回当前库存
$sell = static function() {echo "当前库存为". --static::$num,"\n" ;
};// 将静态匿名函数绑定到 Good 类中
$sold = \Closure::bind($sell, null, Good::class);$sold(); 
$sold();

输出如下

当前库存为 9
当前库存为 8

2.4 call

call 方法是PHP 7 新增的,可以实现绑定并调用匿名函数,语法更加简洁,性能更高。下面是两种方式的示例代码。

// call 版本
$addDiscount->call($good, 0.5);  // 绑定并传入参数 0.5,结果为 50// bindTo 版本
$count = $addDiscount->bindTo($good, Good::class); 
$count(0.5); // 50

2.5 fromCallable

将给定的 callable 函数转化成匿名函数

class Good {private $price;public function __construct(float $price){$this->price = $price;}
}function addDiscount(float $discount = 0.8){return $this->price * $discount;
}$closure = \Closure::fromCallable('addDiscount');
$good = new Good(100);
$count = $closure->bindTo($good);  
$count = $closure->bindTo($good, Good::class);   // 报错,不能重复绑定作用域
$count(); // 报错,无法访问私有属性// fromCallable 等价于
$reflexion = new ReflectionFunction('addDiscount');
$closure = $reflexion->getClosure();

有一点需要特别注意,无论是 fromCallable 转化成的闭包,还是使用反射得到的闭包,在使用 bindTo 时,如果第二个参数指定绑定类,就会报下面的错错误

Cannot rebind scope of closure created by ReflectionFunctionAbstract::getClosure()

三、使用

匿名函数的本质是对象,因此可将匿名函数赋值给某一变量。

3.1 使用外部变量

通过 use 声明的变量可以在匿名函数内部使用,如果声明的是引用,那么在匿名函数中修改变量后,外部变量会同步修改。

$num = 1;
$func = function() use($num){$num = $num + 1;echo $num;
}
$func();  // 2
echo $num;  // 还是 1// 要让匿名函数中对外部变量的修改在闭包结束后仍然生效,需要使用引用传值$num = 1;
$funcRef = function() use(&$num){$num = $num + 1;echo $num;
}
$funcRef();  // 2
echo $num;  // 2

3.2 自动绑定$this到当前类

从 PHP 5.4 开始,在类里面使用匿名函数时,匿名函数的 $this 将自动绑定到当前类

class Foo {public function bar(){return function() {return $this;};}public function getAge(){return 100;}
}function testThis(){$foo = new Foo();$obj = $foo->bar();var_dump($obj);//var_dump($obj()->getAge());
}testThis();

输出结果如下

object(Closure)#2 (1) {["this"]=>object(Foo)#1 (0) {}
}
int(100)

如果不想让自动绑定生效,可以使用静态匿名函数,此时返回的闭包中不能使用 $this,否则一调用就会报下面的 Fatal error

class FooStatic
{public function bar(){return static function () {return $this;};}public function getAge(){return 100;}}function testStaticThis()
{$foo = new FooStatic();$obj = $foo->bar(); // Closure()var_dump($obj);$obj(); // Fatal error: Using $this when not in object context
}
testStaticThis();

因此静态匿名函数实际上是绑定了整个类,可以通过static关键字访问类静态变量,示例代码如下

class FooStatic
{static $quantity = 10;public function bar(){return static function () {echo static::$quantity,"\n";};}
}$foo = new FooStatic();
$obj = $foo->bar(); // Closure()
var_dump($obj);$obj();

输出如下

object(Closure)#2 (0) {
}
10

参考

php手册 https://www.php.net/manual/zh/class.closure.php
心智极客 https://learnku.com/articles/35863

相关内容

热门资讯

清朝有很多有名的历史人物,能给... 清朝有很多有名的历史人物,能给大家说几个清朝的历史人物吗?林则徐,张之洞,乾隆皇帝,曾国藩,朱耷,这...
曹丕的“太子四友”指的是谁 曹丕的“太子四友”指的是谁首先说,曹丕这四个,陈群司马懿是顶级的谋士和政治家,吴质有些小聪明,朱铄不...
在这次遇难者中存在了几名幸存者... 在这次遇难者中存在了几名幸存者。这句话是不是逻辑错误这句话的逻辑没有问题,有问题的是用词不当。遇难者...
在中国古代,有许多充满哲学智慧... 在中国古代,有许多充满哲学智慧的成语典故、寓言故事,如...在中国古代,有许多充满哲学智慧的成语典故...
清澈的意思是什么,… 清澈的意思是什么,…清净而明澈清而透明【造句】看着他清澈而又天真的眼眸,我的心久久不能平静……
蚂蚁森林合种爱情树一方退出怎么... 蚂蚁森林合种爱情树一方退出怎么找回来两个人合种的爱情树,我退出来,我怎么再次加入进去继续合作那个树?...
有好看的古代修炼小说推荐吗? 有好看的古代修炼小说推荐吗?古代重生穿越修炼......让我帮你找一下这些古代修炼的小说,找到这些类...
女主重生爱上前世辜负的人 女主重生爱上前世辜负的人重生我是你正妻渣女重生之竹马重生之弃渣重生之夫君可欺重生之换我疼你重生妇归来...
华胥引有广播剧吗 华胥引有广播剧吗现有的华胥引的广播剧是忆语广播剧社出品的,只有十三月和杯(这个是错字,请无视,居然输...
选文韩麦尔先生在说了,我的朋友... 选文韩麦尔先生在说了,我的朋友们我就要离开你们呢了,再见了银头鲑鱼tj75rt6yturdrruv ...
中通快递从北京保定市到广东揭阳... 中通快递从北京保定市到广东揭阳普宁要多久?中通快递从北京保定市到广东揭阳普宁要多久?从北京到广东需要...
关于离婚悲伤的歌曲 关于离婚悲伤的歌曲关于离婚悲伤的歌曲林俊杰《可惜没如果》 张靓颖《我走以后》 金志文《流着泪说分手》...
形容文笔差怎么说啊? 形容文笔差怎么说啊?哥哥姐姐,麻烦问下,我是做文员的,形容文案方面的工作很差应该怎么说啊?粗鄙怎么样...
我想做未婚妈妈,可行吗? 我想做未婚妈妈,可行吗?没关系吧?我同学好多他们妈妈都是30岁以后省得他们,都没事啊,但是如果你自己...
如何在两个excel表格里筛选... 如何在两个excel表格里筛选出重复的名字如何在两个excel表格里筛选出重复的名字1、电脑打开EX...
且试天下 哪些小说是用白绫做武... 且试天下 哪些小说是用白绫做武器的?神雕侠侣吖- -..小龙女一开始就是用白绫的聊斋 辛十四娘嘿嘿~...
自带高冷体质,笑起来温暖又治愈... 自带高冷体质,笑起来温暖又治愈的星座,你了解吗?虽然天生高冷体质,但是笑起来特别的温暖治愈的新作用天...
火星未解之谜 火星未解之谜多列举一些,每个事例最好长一点,谢啦~“火星人脸”, “地表被水冲击河道”,“原始大气和...
我是1991年10月4号生的,... 我是1991年10月4号生的,是什么星座啊有的说是处女座有的说是天平座,糊涂了,到底是什么啊很负责任...
梦见白狐狸,然后当时我骑着自行... 梦见白狐狸,然后当时我骑着自行车,我想躲开它,他很凶的的追赶我,最后它向我扑了过来,然后我就醒乐.你...