Closure::bind

(PHP 5 >= 5.4.0, PHP 7, PHP 8)

Closure::bind Дублирует замыкание с конкретным связанным объектом и областью видимости класса

Описание

public static Closure::bind(Closure $closure, ?object $newThis, object|string|null $newScope = "static"): ?Closure

Этот метод — статическая версия метода Closure::bindTo(). Описание нестатической версии метода даёт дополнительную информацию.

Список параметров

closure

Анонимная функция, которую метод свяжет с объектом.

newThis

Объект, с которым метод свяжет анонимную функцию, или null, чтобы замыкание оставалось несвязанным.

newScope

При передаче строкового аргумента: название класса, с областью видимости которого метод свяжет замыкание, или ключевое слово 'static', чтобы замыкание сохранило текущую область видимости. При передаче объекта как аргумента: роль названия класса будет играть тип объекта. Параметр определяет видимость защищённых и закрытых методов связанного объекта. Нельзя передавать в параметр название или экземпляр объекта внутреннего PHP-класса.

Возвращаемые значения

Метод возвращает новый объект замыкания Closure или null, если возникла ошибка.

Примеры

Пример #1 Пример использования метода Closure::bind()

<?php

class A
{
private static
$sfoo = 1;
private
$ifoo = 2;
}

$cl1 = static function() {
return
A::$sfoo;
};

$cl2 = function() {
return
$this->ifoo;
};

$bcl1 = Closure::bind($cl1, null, 'A');
$bcl2 = Closure::bind($cl2, new A(), 'A');

echo
$bcl1(), "\n";
echo
$bcl2(), "\n";

?>

Вывод приведённого примера будет похож на:

1
2

Смотрите также

Добавить

Примечания пользователей 2 notes

up
95
Vincius Krolow
11 years ago
With this class and method, it's possible to do nice things, like add methods on the fly to an object.

MetaTrait.php
<?php
trait MetaTrait
{

private
$methods = array();

public function
addMethod($methodName, $methodCallable)
{
if (!
is_callable($methodCallable)) {
throw new
InvalidArgumentException('Second param must be callable');
}
$this->methods[$methodName] = Closure::bind($methodCallable, $this, get_class());
}

public function
__call($methodName, array $args)
{
if (isset(
$this->methods[$methodName])) {
return
call_user_func_array($this->methods[$methodName], $args);
}

throw
RunTimeException('There is no method with the given name to call');
}

}
?>

test.php
<?php
require 'MetaTrait.php';

class
HackThursday {
use
MetaTrait;

private
$dayOfWeek = 'Thursday';

}

$test = new HackThursday();
$test->addMethod('when', function () {
return
$this->dayOfWeek;
});

echo
$test->when();

?>
up
9
potherca at hotmail dot com
9 years ago
If you need to validate whether or not a closure can be bound to a PHP object, you will have to resort to using reflection.

<?php

/**
* @param \Closure $callable
*
* @return bool
*/
function isBindable(\Closure $callable)
{
$bindable = false;

$reflectionFunction = new \ReflectionFunction($callable);
if (
$reflectionFunction->getClosureScopeClass() === null
|| $reflectionFunction->getClosureThis() !== null
) {
$bindable = true;
}

return
$bindable;
}
?>
To Top