В php замыкания это объекты класса Сlosure, создаются при присвоении функции переменной или передачи анонимной функции, и может замыкать в себе значения переменных.
В отличии от JS, функции в PHP не имеют доступа к значениям переменных из родительской области видимости автоматически, его можно получить через use()
Если при создании замыкания мы наследуем переменные из глобальной области через use(), то значение этих переменных будет хранится в замыкании, и будут соответствовать значениям переменных на момент определения функции, а не на момент её вызова.
$message = 'привет';
// Наследуем $message
$example = function () use ($message) {
var_dump($message);
};
$example(); // "Привет"
// Значение унаследованной переменной задано там, где функция определена,
// но не там, где вызвана (в отличии от JS)
$message = 'мир';
$example(); // "Привет"
Так же созданное замыкание может хранить в себе значение переменной...
function getCounter(){
$count = 0;
return function () use (&$count){ // наследуем по ссылке $count из родительской области видимости
return $count++;
};
}
$counter = getCounter(); // получаем замыкание которое хранит в себе значение переменной $count, которая находится в его "замкнутой" родительской области видимости
var_dump($counter);
echo $counter(); // 0
echo $counter(); // 1
echo $counter(); // 2
var_dump($counter);
Замыкания в PHP в ООП (Автоматическое связывание $this)
В ООП, когда мы записываем метод обьекта в функцию (например ), то получаем замыкание, которое хранит контекст и значение $this соответствует объекту из которого мы записывали метод.
Но его можно привязать к другому объекту функцией bindTo();
class Person {
public $name;
public function __construct($name){
$this->name = $name;
}
public function getName(){
return $this->name;
}
public function runClosure($closure){
// возвращаем результат функции переданной в качестве аргумента
return $closure();
}
}
$gizmo = new Person('Gizmo','mr.');
$freeman = new Person('Freeman', 'dr.');
// записываем метод getName() в переменную (именно объект Closure с функцией внутри, а не результат функции)
$closure = $gizmo->getName(...);
// вызывем замыкание внутри объекта $freeman, но оно вернёт "Gizmo" так как само замыкание хранит $this которое ссылается на свой объект, в данном случае $gizmo
echo $freeman->runClosure($closure);
// связываем замыкание с объектом $freeman
$closure = $closure->bindTo($freeman);
// и теперь замыкание выдаст "Freeman"
echo $freeman->runClosure($closure);