在旧版的 PHP 中,您可以通过 create_function,以编程方式创建函数,它们允许为传递功能提供一个解决方法:以字符串的形式发送函数的名称,然后通过 call_user_func 和 call_user_func_array 来调用函数。这种选择缺乏真正匿名函数的优雅,真正的匿名函数可以在方法和类之间传递函数名称,或者在适当的范围内将它们保存在变量中。
匿名函数在 JavaScript 中无处不在,不知道 JavaScript 的 PHP 程序员很少见。所以扩展 PHP 来使之包含匿名函数是自然而然的事情。自 PHP 5.3 起,您可以在能够使用变量(用于存储或传递)的任何地方使用普通的函数声明语法。
作为一个示例,下面的代码显示了以前使用内置排序函数来指定您自己的自定义排序功能的方式。
以前使用的方式
<?php
$insurees = [
'u4937' => ['name' => 'Thomas Smythe', 'age' => 33],
'u1282' => ['name' => 'Gayle Runecor', 'age' => 25],
'u9275' => ['name' => 'Sara Pinnicle', 'age' => 57],
'u2078' => ['name' => 'Delilah Shock', 'age' => 41],
];
function insuree_age_sort($a, $b) {
if ($a['age'] == $b['age']) { return 0; }
return ($a['age'] > $b['age']) ? -1 : 1;
}
uasort($insurees, 'insuree_age_sort');
上面的代码有一些繁琐,因为您需要在相同的范围内定义一个函数,然后使用它(即使您永远不会再使用它)。通过使用闭包,您现在就可以只使用一个步骤来直接创建和使用该函数。清单 7 显示了一个这样的示例,这是一个更优雅的解决方案
使用一个匿名函数来实现排序
<?php
uasort($insurees, function ($a, $b) {
if ($a['age'] == $b['age']) { return 0; }
return ($a['age'] > $b['age']) ? -1 : 1;
});
不过,有人宣称这小的用例是提供此特性的惟一理由。但意识到这里发生的一切之后,我发现,我可以动态创建函数,并将它传入 uasort(),后者是一个一级的变量。您可以在变量中存储函数,并将它们传递给不同的函数和类。当您看到被添加到 PHP 和闭包中的作用域特性时,闭包的真正力量就会变得显而易见。
借助被广使用的 use 关键字,您可以在函数有权访问的某个作用域内指定某些变量。通过这种方式,您可以处理相当复杂的一些细节,而不必在每次以变量形式访问函数时不断地将这些细节传递到函数中。清单 8 和清单 9 中(人为设计)的示例展示了这种力量。
在回调中使用继承的变量作用域
<?php
// Find only people over a certain age
$minage = 40;
$over = array_filter($insurees, function($a) use ($minage) {
return ($a['age'] >= $minage);
});
包含多个变量和直接调用的闭包
<?php
$urls = [
'training' => '/training',
'magazine' => '/magazine',
't-shirt' => '/swag/tshirts',
];
$current = $_SERVER['REQUEST_URI']; // May come from somewhere else
// Helper for links, ignoring links if we are on that page:
$link = function($name) use ($urls, $current) {
if ($current == $urls[$name]) {
return $name;
} else {
return "<a href=\"{$urls[$name]}\">{$name}</a>";
}
};
?>
<p>Welcome to our website! Make sure to check out
our <?= $link('training') ?> offerings, see the
latest issue of our <?= $link('magazine'); ?>,
and don't forget to check out our latest
<?= $link('t-shirt') ?> designs as well.</p>
如果您习惯在 JavaScript 中使用闭包,那么您应该已经熟悉它们的能力、灵活性和有时有点危险的特性。