2024 年 PHP 日本大会

匿名函数

匿名函数,也称为闭包,允许创建没有指定名称的函数。它们最常用于可调用参数的值,但它们还有许多其他用途。

匿名函数使用 Closure 类实现。

示例 #1 匿名函数示例

<?php
echo preg_replace_callback('~-([a-z])~', function ($match) {
return
strtoupper($match[1]);
},
'hello-world');
// 输出 helloWorld
?>

闭包也可以用作变量的值;PHP 会自动将此类表达式转换为Closure 内部类的实例。将闭包赋给变量使用与其他任何赋值相同的语法,包括尾随分号。

示例 #2 匿名函数变量赋值示例

<?php
$greet
= function($name) {
printf("Hello %s\r\n", $name);
};

$greet('World');
$greet('PHP');
?>

闭包还可以从父作用域继承变量。任何此类变量都必须传递给use语言结构。从 PHP 7.1 开始,这些变量不得包含超全局变量$this或与参数同名的变量。函数的返回类型声明必须放在use子句之后。

示例 #3 从父作用域继承变量

<?php
$message
= 'hello';

// 没有 "use"
$example = function () {
var_dump($message);
};
$example();

// 继承 $message
$example = function () use ($message) {
var_dump($message);
};
$example();

// 继承的变量值来自定义函数时,而不是调用时
$message = 'world';
$example();

// 重置 message
$message = 'hello';

// 通过引用继承
$example = function () use (&$message) {
var_dump($message);
};
$example();

// 父作用域中更改的值
// 反映在函数调用中
$message = 'world';
$example();

// 闭包也可以接受常规参数
$example = function ($arg) use ($message) {
var_dump($arg . ' ' . $message);
};
$example("hello");

// 返回类型声明位于 use 子句之后
$example = function () use ($message): string {
return
"hello $message";
};
var_dump($example());
?>

以上示例将输出类似于以下内容:

Notice: Undefined variable: message in /example.php on line 6
NULL
string(5) "hello"
string(5) "hello"
string(5) "hello"
string(5) "world"
string(11) "hello world"
string(11) "hello world"

从 PHP 8.0.0 开始,作用域继承变量列表可以包含尾随逗号,该逗号将被忽略。

从父作用域继承变量与使用全局变量**不同**。全局变量存在于全局作用域中,无论执行哪个函数,全局作用域都是相同的。闭包的父作用域是声明闭包的函数(不一定是从其调用的函数)。请参见以下示例

示例 #4 闭包和作用域

<?php
// 一个基本的购物车,包含已添加产品的列表
// 和每种产品的数量。包含一个方法,使用
// 闭包作为回调来计算购物车中商品的总价。
class Cart
{
const
PRICE_BUTTER = 1.00;
const
PRICE_MILK = 3.00;
const
PRICE_EGGS = 6.95;

protected
$products = array();

public function
add($product, $quantity)
{
$this->products[$product] = $quantity;
}

public function
getQuantity($product)
{
return isset(
$this->products[$product]) ? $this->products[$product] :
FALSE;
}

public function
getTotal($tax)
{
$total = 0.00;

$callback =
function (
$quantity, $product) use ($tax, &$total)
{
$pricePerItem = constant(__CLASS__ . "::PRICE_" .
strtoupper($product));
$total += ($pricePerItem * $quantity) * ($tax + 1.0);
};

array_walk($this->products, $callback);
return
round($total, 2);
}
}

$my_cart = new Cart;

// 向购物车添加一些商品
$my_cart->add('butter', 1);
$my_cart->add('milk', 3);
$my_cart->add('eggs', 6);

// 打印包含5%销售税的总价。
print $my_cart->getTotal(0.05) . "\n";
// 结果是 54.29
?>

示例 #5 自动绑定 $this

<?php

class Test
{
public function
testing()
{
return function() {
var_dump($this);
};
}
}

$object = new Test;
$function = $object->testing();
$function();

?>

以上示例将输出

object(Test)#1 (0) {
}

在类上下文中声明时,当前类会自动绑定到它,使 $this 在函数作用域内可用。如果不需要这种当前类的自动绑定,则可以使用静态匿名函数

静态匿名函数

匿名函数可以声明为静态的。这可以防止它们自动绑定当前类。对象也不能在运行时绑定到它们。

示例 #6 尝试在静态匿名函数内使用 $this

<?php

class Foo
{
function
__construct()
{
$func = static function() {
var_dump($this);
};
$func();
}
};
new
Foo();

?>

以上示例将输出

Notice: Undefined variable: this in %s on line %d
NULL

示例 #7 尝试将对象绑定到静态匿名函数

<?php

$func
= static function() {
// 函数体
};
$func = $func->bindTo(new stdClass);
$func();

?>

以上示例将输出

Warning: Cannot bind an instance to a static closure in %s on line %d

变更日志

版本 描述
8.3.0 魔术方法创建的闭包可以接受命名参数。
7.1.0 匿名函数不能闭包超全局变量$this或任何与参数同名的变量。

备注

注意: 可以在闭包内使用func_num_args()func_get_arg()func_get_args()

添加备注

用户贡献的笔记 20 条笔记

orls
14 年前
注意将变量“导入”到闭包的作用域时——很容易错过/忘记它们实际上是*复制*到闭包的作用域中,而不是仅仅使其可用。

因此,如果您的闭包关心它们随时间的变化,则需要显式地通过引用传递它们。

<?php
$result
= 0;

$one = function()
{
var_dump($result); };

$two = function() use ($result)
{
var_dump($result); };

$three = function() use (&$result)
{
var_dump($result); };

$result++;

$one(); // 输出 NULL: $result 不在作用域内
$two(); // 输出 int(0): $result 已被复制
$three(); // 输出 int(1)
?>

另一个关于对象的不太简单的示例(我实际遇到的问题)

<?php
//预先设置变量
$myInstance = null;

$broken = function() uses ($myInstance)
{
if(!empty(
$myInstance)) $myInstance->doSomething();
};

$working = function() uses (&$myInstance)
{
if(!empty(
$myInstance)) $myInstance->doSomething();
}

//$myInstance 可能已实例化,也可能未实例化
if(SomeBusinessLogic::worked() == true)
{
$myInstance = new myClass();
}

$broken(); // 永远不会执行任何操作:在此闭包内 $myInstance 将始终为 null。
$working(); // 如果 $myInstance 已实例化,则会调用 doSomething

?>
[email protected]
9年前
<?php
/*
(string) $name 你将添加到类的函数名称。
用法:$Foo->add(function(){},$name);
这将在 Foo 类中添加一个公共函数。
*/
class Foo
{
public function
add($func,$name)
{
$this->{$name} = $func;
}
public function
__call($func,$arguments){
call_user_func_array($this->{$func}, $arguments);
}
}
$Foo = new Foo();
$Foo->add(function(){
echo
"Hello World";
},
"helloWorldFunction");
$Foo->add(function($parameterone){
echo
$parameterone;
},
"exampleFunction");
$Foo->helloWorldFunction(); /*输出:Hello World*/
$Foo->exampleFunction("Hello PHP"); /*输出:Hello PHP*/
?>
cHao
10年前
如果你想知道(因为我当时想知道),匿名函数可以像命名函数一样返回引用。只需像使用命名函数一样使用 &…在 `function` 关键字之后(以及不存在的名称之前)。

<?php
$value
= 0;
$fn = function &() use (&$value) { return $value; };

$x =& $fn();
var_dump($x, $value); // 'int(0)', 'int(0)'
++$x;
var_dump($x, $value); // 'int(1)', 'int(1)'
Hayley Watson
6个月前
如果你在对象属性中存储了闭包(或其他可调用对象),并且想要调用它,可以使用括号来区分它和方法调用。

<?php
class Test
{
public
$callable;

function
__construct()
{
$this->callable = function($a) { return $a + 2; };
}
}

$t = new Test;

echo (
$t->callable)(40);
?>
[email protected]
6年前
lambda 表达式的每个实例都有其自己的静态变量实例。这非常适合事件处理程序、累加器等。

使用 `function() { ... };` 表达式创建新的 lambda 会创建其静态变量的新实例。将 lambda 分配给变量不会创建新实例。lambda 是 Closure 类的对象,将 lambda 分配给变量的语义与将对象实例分配给变量的语义相同。

示例脚本:$a 和 $b 具有单独的静态变量实例,因此产生不同的输出。但是 $b 和 $c 共享它们的静态变量实例——因为 $c 引用与 $b 相同的 Closure 类对象——因此产生相同的输出。

#!/usr/bin/env php
<?php

function generate_lambda() : Closure
{
# 创建 lambda 的新实例
return function($v = null) {
static
$stored;
if (
$v !== null)
$stored = $v;
return
$stored;
};
}

$a = generate_lambda(); # 创建静态变量的新实例
$b = generate_lambda(); # 创建静态变量的新实例
$c = $b; # 使用与 $b 相同的静态变量实例

$a('test AAA');
$b('test BBB');
$c('test CCC'); # 这会覆盖 $b 保持的内容,因为它引用的是同一个对象

var_dump([ $a(), $b(), $c() ]);
?>

此测试脚本输出
array(3) {
[0]=>
string(8) "test AAA"
[1]=>
string(8) "test CCC"
[2]=>
string(8) "test CCC"
}
[email protected]
15年前
在类中使用匿名函数作为属性时,请注意存在三个命名范围:一个用于常量,一个用于属性,一个用于方法。这意味着你可以同时使用相同的名称来表示常量、属性和方法。

由于从 PHP 5.3.0 开始,属性也可以是匿名函数,因此当它们共享相同的名称时会出现一个奇特的情况,但这并不意味着会发生任何冲突。

考虑以下示例

<?php
class MyClass {
const
member = 1;

public
$member;

public function
member () {
return
"method 'member'";
}

public function
__construct () {
$this->member = function () {
return
"anonymous function 'member'";
};
}
}

header("Content-Type: text/plain");

$myObj = new MyClass();

var_dump(MyClass::member); // int(1)
var_dump($myObj->member); // object(Closure)#2 (0) {}
var_dump($myObj->member()); // string(15) "method 'member'"
$myMember = $myObj->member;
var_dump($myMember()); // string(27) "anonymous function 'member'"
?>

这意味着常规方法调用按预期和以前一样工作。相反,匿名函数必须首先被检索到一个变量中(就像属性一样),然后才能被调用。

此致,
[email protected]
7年前
递归调用匿名函数的一种方法是使用 USE 关键字并传递对函数本身的引用。

<?php
$count
= 1;
$add = function($count) use (&$add){
$count += 1;
if(
$count < 10) $count = $add($count); //递归调用
return $count;
};
echo
$add($count); //将输出10,符合预期
?>
rob at ubrio dot us
15年前
您可以始终使用`__call()`方法调用受保护的成员 - 这类似于您在 Ruby 中使用`send`绕过此问题的方式。

<?php

class Fun
{
protected function
debug($message)
{
echo
"DEBUG: $message\n";
}

public function
yield_something($callback)
{
return
$callback("Soemthing!!");
}

public function
having_fun()
{
$self =& $this;
return
$this->yield_something(function($data) use (&$self)
{
$self->debug("Doing stuff to the data");
// 对 $data 做一些操作
$self->debug("Finished doing stuff with the data.");
});
}

// 啊哈!
public function __call($method, $args = array())
{
if(
is_callable(array($this, $method)))
return
call_user_func_array(array($this, $method), $args);
}
}

$fun = new Fun();
echo
$fun->having_fun();

?>
mail at mkharitonov dot net
10年前
PHP 和 JavaScript 闭包的一些比较。

=== 示例 1(按值传递)===
PHP 代码
<?php
$aaa
= 111;
$func = function() use($aaa){ print $aaa; };
$aaa = 222;
$func(); // 输出 "111"
?>

类似的 JavaScript 代码
<script type="text/javascript">
var aaa = 111;
var func = (function(aaa){ return function(){ alert(aaa); } })(aaa);
aaa = 222;
func(); // 输出 "111"
</script>

小心,下面的代码与之前的代码不同
<script type="text/javascript">
var aaa = 111;
var bbb = aaa;
var func = function(){ alert(bbb); };
aaa = 222;
func(); // 输出 "111",但仅在函数声明后 "bbb" 未更改时

// 此技术不适用于循环
var functions = [];
for (var i = 0; i < 2; i++)
{
var i2 = i;
functions.push(function(){ alert(i2); });
}
functions[0](); // 输出 "1",错误!
functions[1](); // 输出 "1",正确
</script>

=== 示例 2(按引用传递)===
PHP 代码
<?php
$aaa
= 111;
$func = function() use(&$aaa){ print $aaa; };
$aaa = 222;
$func(); // 输出 "222"
?>

类似的 JavaScript 代码
<script type="text/javascript">
var aaa = 111;
var func = function(){ alert(aaa); };
aaa = 222; // 输出 "222"
func();
</script>
simon at generalflows dot com
13 年前
<?php

/*
* 一个示例,展示如何使用闭包来实现类似 Python 的装饰器
* 模式。
*
* 我的目标是您应该能够用任何其他函数来装饰一个函数,然后直接调用被装饰的函数:
*
* 定义函数:$foo = function($a, $b, $c, ...) {...}
* 定义装饰器:$decorator = function($func) {...}
* 装饰它:$foo = $decorator($foo)
* 调用它:$foo($a, $b, $c, ...)
*
* 此示例显示了使用简单的模拟会话和模拟服务的身份验证装饰器。
*/

session_start();

/*
* 定义一个示例装饰器。装饰器函数应采用以下形式:
* $decorator = function($func) {
* return function() use $func) {
* // 做一些事情,然后在需要时调用被装饰的函数:
* $args = func_get_args($func);
* call_user_func_array($func, $args);
* // 做其他事情。
* };
* };
*/
$authorise = function($func) {
return function() use (
$func) {
if (
$_SESSION['is_authorised'] == true) {
$args = func_get_args($func);
call_user_func_array($func, $args);
}
else {
echo
"Access Denied";
}
};
};

/*
* 定义一个要装饰的函数,在此示例中,这是一个需要授权的模拟服务。
*/
$service = function($foo) {
echo
"Service returns: $foo";
};

/*
* 装饰它。确保用装饰后的函数替换原始函数引用;即仅 $authorise($service) 不起作用,因此请执行
* $service = $authorise($service)
*/
$service = $authorise($service);

/*
* 建立模拟授权,调用服务;应该得到
* 'Service returns: test 1'。
*/
$_SESSION['is_authorised'] = true;
$service('test 1');

/*
* 删除模拟授权,调用服务;应该得到 'Access Denied'。
*/
$_SESSION['is_authorised'] = false;
$service('test 2');

?>
john at binkmail dot com
7年前
2017 年性能基准测试!

我决定将单个保存的闭包与在每次循环迭代中不断创建相同的匿名闭包进行比较。我尝试了 1000 万次循环迭代,使用的是 2016 年 12 月的 PHP 7.0.14 版本。结果

单个保存在变量中并重复使用的闭包(10000000 次迭代):1.3874590396881 秒

每次都创建一个新的匿名闭包(10000000 次迭代):2.8460240364075 秒

换句话说,在 1000 万次迭代的过程中,每次迭代都重新创建闭包总共只增加了“1.459 秒”的运行时间。这意味着在我的 7 年历史的双核笔记本电脑上,每次创建新的匿名闭包大约需要 146 纳秒。我猜 PHP 会为匿名函数保留一个缓存的“模板”,因此不需要太多时间来创建闭包的新实例!

因此,您不必担心在紧密循环中不断地一遍又一遍地重新创建匿名闭包!至少在 PHP 7 中是这样!绝对不需要将实例保存在变量中并重复使用它。不受此限制是一件好事,因为它意味着您可以随意地在闭包有意义的地方使用它们,而不是在代码的其他地方定义它们。 :-)
derkontrollfreak+9hy5l at gmail dot com
10年前
请注意,从 PHP 5.4 开始,将闭包注册为在同一对象作用域中实例化的对象属性将创建一个循环引用,从而阻止对象立即销毁。
<?php

class Test
{
private
$closure;

public function
__construct()
{
$this->closure = function () {
};
}

public function
__destruct()
{
echo
"destructed\n";
}
}

new
Test;
echo
"finished\n";

/*
* PHP 5.3 中的结果:
* ------------------
* destructed
* finished
*
* 从 PHP 5.4 开始的结果:
* ---------------------
* finished
* destructed
*/

?>

为避免这种情况,您可以在静态方法中实例化闭包。
<?php

public function __construct()
{
$this->closure = self::createClosure();
}

public static function
createClosure()
{
return function () {
};
}

?>
toonitw at gmail dot com
6年前
从 PHP 7.0 开始,您可以使用 IIFE(立即调用函数表达式)将匿名函数用 () 包裹。

<?php
$type
= 'number';
var_dump( ...( function() use ($type) {
if (
$type=='number') return [1,2,3];
else if (
$type=='alphabet') return ['a','b','c'];
} )() );
?>
jake dot tunaley at berkeleyit dot com
5年前
小心在赋值给静态变量的匿名函数中使用 `$this`。

<?php
class Foo {
public function
bar() {
static
$anonymous = null;
if (
$anonymous === null) {
// 表达式不允许作为静态初始化的变通方法
$anonymous = function () {
return
$this;
};
}
return
$anonymous();
}
}

$a = new Foo();
$b = new Foo();
var_dump($a->bar() === $a); // True
var_dump($b->bar() === $a); // 同样为 True
?>

在静态匿名函数中,`$this` 将是首次调用该方法的对象实例的值。

要获得你可能期望的行为,你需要将 `$this` 上下文传递到函数中。

<?php
class Foo {
public function
bar() {
static
$anonymous = null;
if (
$anonymous === null) {
// 表达式不允许作为静态初始化的变通方法
$anonymous = function (self $thisObj) {
return
$thisObj;
};
}
return
$anonymous($this);
}
}

$a = new Foo();
$b = new Foo();
var_dump($a->bar() === $a); // True
var_dump($b->bar() === $a); // False
?>
Hayley Watson
1年前
"如果不需要这种当前类的自动绑定,则可以使用静态匿名函数代替。"

你不希望自动绑定的主要原因是:只要为匿名函数创建的闭包对象存在,它就会保留对生成它的对象的引用,阻止该对象被销毁,即使该对象在程序的其他任何地方都不再存在,即使函数本身不使用 `$this`。

<?php

class Foo
{
public function
__construct(private string $id)
{
echo
"Creating Foo " . $this->id, "\n";
}
public function
gimme_function()
{
return function(){};
}
public function
gimme_static_function()
{
return static function(){};
}
public function
__destruct()
{
echo
"Destroying Foo " . $this->id, "\n";
}
}

echo
"An object is destroyed as soon as its last reference is removed.\n";
$t = new Foo('Alice');
$t = new Foo('Bob'); // 导致 Alice 被销毁。
// 现在销毁 Bob。
unset($t);
echo
"---\n";

echo
"A non-static anonymous function retains a reference to the object which created it.\n";
$u = new Foo('Carol');
$ufn = $u->gimme_function();
$u = new Foo('Daisy'); // 不会导致 Carol 被销毁,
// 因为函数 $ufn 中仍然存在对它的引用。
unset($u); // 销毁 Daisy。
echo "---\n"; // 注意 Carol 尚未被销毁。

echo "A static anonymous function does not retain a reference to the object which created it.\n";
$v = new Foo('Eve');
$vfn = $v->gimme_static_function();
$v = new Foo('Farid'); // $vfn 中保存的函数
// 不保留对 Eve 的引用,因此 Eve 在此处确实会被销毁。
unset($v); // 销毁 Farid
echo "---\n";
// 然后程序结束,丢弃对任何仍然存在的对象的任何引用
// (特别是 Carol)。
?>

因为 `$ufn` 在程序结束时仍然存在,所以 Carol 也仍然存在。`$vfn` 也在程序结束时仍然存在,但它包含的函数声明为静态的,因此没有保留对 Eve 的引用。

保留对已死对象的引用的匿名函数因此是潜在的内存泄漏来源。如果函数不需要生成它的对象,则将其声明为静态可以防止它导致对象超出其使用期限。
kdelux at gmail dot com
14 年前
这是一个定义然后在闭包函数中使用变量(`$this`)的一种方法的示例。下面的代码探讨了所有用法,并展示了限制。

此代码片段中最有用的工具是 `requesting_class()` 函数,它将告诉您哪个类负责执行当前的 Closure()。

概述
-----------------------
成功找到调用对象的引用。
成功调用 `$this(__invoke);`
成功引用 `$this->name;`
成功调用 `call_user_func(array($this, 'method'))`

失败:通过 `$this->` 引用任何内容
失败:`$this->name = '';`
失败:`$this->delfect();`

<?php



function requesting_class()
{
foreach(
debug_backtrace(true) as $stack){
if(isset(
$stack['object'])){
return
$stack['object'];
}
}

}






class
Person
{
public
$name = '';
public
$head = true;
public
$feet = true;
public
$deflected = false;

function
__invoke($p){ return $this->$p; }
function
__toString(){ return 'this'; } // 测试引用

function __construct($name){ $this->name = $name; }
function
deflect(){ $this->deflected = true; }

public function
shoot()
{
// 如果定义了 customAttack,则将其用作射击结果。否则射击脚
if(is_callable($this->customAttack)){
return
call_user_func($this->customAttack);
}

$this->feet = false;
}
}

$p = new Person('Bob');


$p->customAttack =
function(){

echo
$this; // Notice: Undefined variable: this

#$this = new Class() // 严重错误

// 技巧:赋值变量 '$this'
extract(array('this' => requesting_class())); // 确定哪个类负责调用闭包

var_dump( $this ); // 被动引用有效
var_dump( $$this ); // 添加到类中:function __toString(){ return 'this'; }

$name = $this('name'); // 成功
echo $name; // 输出:Bob
echo '<br />';
echo $
$this->name;

call_user_func_array(array($this, 'deflect'), array()); // 成功调用

#$this->head = 0; //** 严重错误:在非对象上下文中使用 $this
$$this->head = 0; // 成功设置值

};

print_r($p);

$p->shoot();

print_r($p);


die();

?>
mike at borft dot student dot utwente dot nl
12年前
既然可以将闭包赋值给类变量,很可惜无法直接调用它们。例如,以下代码无效
<?php
class foo {

public
test;

public function
__construct(){
$this->test = function($a) {
print
"$a\n";
};
}
}

$f = new foo();

$f->test();
?>

但是,可以使用魔术方法 __call 函数
<?php
class foo {

public
test;

public function
__construct(){
$this->test = function($a) {
print
"$a\n";
};
}

public function
__call($method, $args){
if (
$this->{$method} instanceof Closure ) {
return
call_user_func_array($this->{$method},$args);
} else {
return
parent::__call($method, $args);
}
}
}
$f = new foo();
$f->test();
?>

希望对某些人有所帮助;)
匿名用户
15年前
如果您想专门检查是否正在处理闭包而不是字符串或数组回调,您可以这样做

<?php
$isAClosure
= is_callable($thing) && is_object($thing);
?>
petersenc at gmail dot com
4个月前
从PHP 8.3.9开始,PHP不允许在use语句中进行类型提示。考虑以下Laravel路由

Route::get('/tags/{tag}', function (string $tag) use ($posts): View {
$tagPosts = $posts->filter(
function (Post $post) use ($tag): bool {
return in_array($tag, $post->tags);
}
);

return view('tags.show', [
'posts' => $tagPosts,
'tag' => $tag
]);
});

正如您所看到的,我可以通过对参数和返回类型进行类型提示来使闭包中的代码更加冗长。但是,use不允许类型提示。
gabriel dot totoliciu at ddsec dot net
14 年前
如果您想创建一个递归闭包,您需要这样写

$some_var1="1";
$some_var2="2";

function($param1, $param2) use ($some_var1, $some_var2)
{

//此处添加一些代码

call_user_func(__FUNCTION__, $other_param1, $other_param2);

//此处添加一些代码

}

如果您需要按引用传递值,您应该查看

https://php.net/manual/en/function.call-user-func.php
https://php.net/manual/en/function.call-user-func-array.php

如果您想知道使用call_user_func后$some_var1和$some_var2是否仍然可见,是的,它们是可用的。
To Top