Анонимные функции, также известные как замыкания (closures), позволяют создавать функции, не имеющие определенных имен. Они наиболее полезны в качестве значений callback-параметров, но также могут иметь и множество других применений.
Анонимные функции реализуются с использованием класса 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.
Пример #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';
echo $example();
// Замыкания могут принимать обычные аргументы
$example = function ($arg) use ($message) {
var_dump($arg . ' ' . $message);
};
$example("hello");
?>
Результатом выполнения данного примера будет что-то подобное:
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"
Наследование переменных из родительской области видимости не то же самое, что использование глобальных переменных. Глобальные переменные существуют в глобальной области видимости, которая не меняется, вне зависимости от того, какая функция выполняется в данный момент. Родительская область видимости - это функция, в которой было объявлено замыкание (не обязательно та же самая, из которой оно было вызвано). Смотрите следующий пример:
Пример #4 Замыкания и область видимости
<?php
// Базовая корзина покупок, содержащая список добавленных
// продуктов и количество каждого продукта. Включает метод,
// вычисляющий общую цену элементов корзины с помощью
// callback-замыкания.
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) { }
Результат выполнения данного примера в PHP 5.3:
Notice: Undefined variable: this in script.php on line 8 NULL
Начиная с PHP 5.4.0, при объявлении в контексте класса, текущий класс будет автоматически связан с ним, делая $this доступным внутри функций класса. Если вы не хотите автоматического связывания с текущим классов, используйте статические анонимные функции
Начиная с PHP 5.4, анонимные функции могут быть объявлены статически. Это предотвратит их автоматическое связывание с текущим классом. Объекты также не будут с ними связаны во время выполнения.
Пример #6 Попытка использовать $this в статической анонимной функции
<?php
new class {
function __construct()
{
(static function() {
var_dump($this);
})();
}
};
?>
Результат выполнения данного примера:
Notice: Undefined variable: this in %s on line %d NULL
Пример #7 Попытка связать объект со статической анонимной функцией
<?php
(static function() {
// function body
})->call(new StdClass);
?>
Результат выполнения данного примера:
Warning: Cannot bind an instance to a static closure in /in/sLLS9 on line 5
Версия | Описание |
---|---|
5.4.0 | Стало возможным использовать $this в анонимных функциях, и объявлять их статически. |
5.3.0 | Появление анонимных функций. |
Замечание: Совместно с замыканиями можно использовать функции func_num_args(), func_get_arg() и func_get_args().