Як записуються зворотні дзвінки в 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. Покладатися на цю функцію дуже не рекомендується.