Как обратные вызовы написаны на PHP?
Как обратные вызовы написаны на PHP?
Ответы:
В руководстве термины «обратный вызов» и «вызываемый» используются взаимозаменяемо, однако «обратный вызов» традиционно относится к значению строки или массива, которое действует как указатель на функцию , ссылаясь на метод функции или класса для последующего вызова. Это позволило некоторые элементы функционального программирования начиная с PHP 4. Ароматы:
$cb1 = 'someGlobalFunction';
$cb2 = ['ClassName', 'someStaticMethod'];
$cb3 = [$object, 'somePublicMethod'];
// this syntax is callable since PHP 5.2.3 but a string containing it
// cannot be called directly
$cb2 = 'ClassName::someStaticMethod';
$cb2(); // fatal error
// legacy syntax for PHP 4
$cb3 = array(&$object, 'somePublicMethod');
Это безопасный способ использовать вызываемые значения в целом:
if (is_callable($cb2)) {
// Autoloading will be invoked to load the class "ClassName" if it's not
// yet defined, and PHP will check that the class has a method
// "someStaticMethod". Note that is_callable() will NOT verify that the
// method can safely be executed in static context.
$returnValue = call_user_func($cb2, $arg1, $arg2);
}
Современные версии PHP позволяют напрямую вызывать первые три формата, указанные выше $cb()
. call_user_func
и call_user_func_array
поддерживать все вышеперечисленное.
Смотрите: http://php.net/manual/en/language.types.callable.php
Примечания / Предостережения:
['Vendor\Package\Foo', 'method']
call_user_func
не поддерживает прохождение не-объектов с помощью ссылки, так что вы можете использовать call_user_func_array
или, в более поздних версиях PHP, сохранить обратный вызов в вар и использовать прямой синтаксис: $cb()
;__invoke()
методом (включая анонимные функции) подпадают под категорию «вызываемые» и могут использоваться таким же образом, но я лично не связываю их с устаревшим термином «обратный вызов».create_function()
создает глобальную функцию и возвращает ее имя. Это обертка для eval()
и вместо нее должны использоваться анонимные функции.'someGlobalFunction'
это определенная функция.
С PHP 5.3 вы можете сделать это:
function doIt($callback) { $callback(); }
doIt(function() {
// this will be done
});
Наконец-то хороший способ сделать это. Отличное дополнение к PHP, потому что обратные вызовы потрясающие.
Реализация обратного вызова делается так
// This function uses a callback function.
function doIt($callback)
{
$data = "this is my data";
$callback($data);
}
// This is a sample callback function for doIt().
function myCallback($data)
{
print 'Data is: ' . $data . "\n";
}
// Call doIt() and pass our sample callback function's name.
doIt('myCallback');
Отображает: Данные есть: это мои данные
call_user_func()
Когда у них есть синтаксис, который позволяет им динамически вызывать функции и делать обратные вызовы. Я согласен с тобой!
Недавно я обнаружил один изящный прием - использовать PHP create_function()
для создания анонимной / лямбда-функции для одноразового использования. Это полезно для PHP функций , таких как array_map()
, preg_replace_callback()
, или , usort()
что использование обратных вызовов для таможенного оформления. Он выглядит почти так же, как и eval()
под одеялом, но это все еще хороший способ использовать PHP в функциональном стиле.
create_function()
?
хорошо ... с 5.3 на горизонте все будет лучше, потому что с 5.3 мы получим замыкания и с ними анонимные функции
Вы хотите проверить, что ваш звонок действителен. Например, в случае конкретной функции вы захотите проверить и посмотреть, существует ли функция:
function doIt($callback) {
if(function_exists($callback)) {
$callback();
} else {
// some error handling
}
}
is_callable( $callback )
create_function
не работал для меня в классе. Я должен был использовать call_user_func
.
<?php
class Dispatcher {
//Added explicit callback declaration.
var $callback;
public function Dispatcher( $callback ){
$this->callback = $callback;
}
public function asynchronous_method(){
//do asynch stuff, like fwrite...then, fire callback.
if ( isset( $this->callback ) ) {
if (function_exists( $this->callback )) call_user_func( $this->callback, "File done!" );
}
}
}
Затем использовать:
<?php
include_once('Dispatcher.php');
$d = new Dispatcher( 'do_callback' );
$d->asynchronous_method();
function do_callback( $data ){
print 'Data is: ' . $data . "\n";
}
?>
[Править] Добавлена пропущенная скобка. Кроме того, добавил объявление обратного вызова, я предпочитаю это таким образом.
Я съеживаюсь каждый раз, когда использую create_function()
в php.
Параметры - это строка, разделенная запятой, все тело функции в строке ... Аааа ... Я думаю, что они не могли бы сделать это еще хуже, даже если бы попытались.
К сожалению, это единственный выбор, когда создание именованной функции не стоит хлопот.
create_function()
сейчас устарела и не должна использоваться.
Для тех, кто не заботится о нарушении совместимости с PHP < 5.4
, я бы предложил использовать подсказки типов для более чистой реализации.
function call_with_hello_and_append_world( callable $callback )
{
// No need to check $closure because of the type hint
return $callback( "hello" )."world";
}
function append_space( $string )
{
return $string." ";
}
$output1 = call_with_hello_and_append_world( function( $string ) { return $string." "; } );
var_dump( $output1 ); // string(11) "hello world"
$output2 = call_with_hello_and_append_world( "append_space" );
var_dump( $output2 ); // string(11) "hello world"
$old_lambda = create_function( '$string', 'return $string." ";' );
$output3 = call_with_hello_and_append_world( $old_lambda );
var_dump( $output3 ); // string(11) "hello world"
create_function()
устарело с версии PHP 7.2.0. Полагаться на эту функцию крайне не рекомендуется.