Callback / Callable 类型

回调可以通过 callable 类型声明来表示。

一些函数如 call_user_func()usort() 可以接受用户自定义的回调函数作为参数。回调函数不止可以是简单函数,还可以是对象的方法,包括静态类方法。

传递

PHP是将函数以string形式传递的。 可以使用任何内置或用户自定义函数,但除了语言结构例如:array()echoempty()eval()exit()isset()list()printunset()

一个已实例化的 object 的方法被作为 array 传递,下标 0 包含该 object,下标 1 包含方法名。 在同一个类里可以访问 protected 和 private 方法。

静态类方法可以不实例化 object 传递,只需要在下标为 0 的位置传递类名而不是 object ,或者传递 'ClassName::methodName'

回调参数不仅可以使用普通的用户自定义函数,也接受 匿名函数箭头函数

注意:

从 PHP 8.1.0 开始,还可以使用 First-class 可调用语法 创建匿名函数。

通常情况下,任何实现了 __invoke() 的对象都可以传入回调参数。

示例 #1 回调函数示例

<?php 

// 回调函数示范
function my_callback_function() {
    echo 
'hello world!';
}

// 回调方法示范
class MyClass {
    static function 
myCallbackMethod() {
        echo 
'Hello World!';
    }
}

// 类型 1:简单的回调
call_user_func('my_callback_function'); 

// 类型 2:静态类方法回调
call_user_func(array('MyClass''myCallbackMethod')); 

// 类型 3:对象方法回调
$obj = new MyClass();
call_user_func(array($obj'myCallbackMethod'));

// 类型 4:静态类方法回调
call_user_func('MyClass::myCallbackMethod');

// 类型 5:父级静态类回调
class {
    public static function 
who() {
        echo 
"A\n";
    }
}

class 
extends {
    public static function 
who() {
        echo 
"B\n";
    }
}

call_user_func(array('B''parent::who')); // A

// 类型 6:实现 __invoke 的对象用于回调
class {
    public function 
__invoke($name) {
        echo 
'Hello '$name"\n";
    }
}

$c = new C();
call_user_func($c'PHP!');
?>

示例 #2 使用 Closure 的示例

<?php
// 闭包
$double = function($a) {
    return 
$a 2;
};

// 这是数字范围
$numbers range(15);

// 这里使用闭包作为回调,
// 将范围内的每个元素数值翻倍
$new_numbers array_map($double$numbers);

print 
implode(' '$new_numbers);
?>

以上例程会输出:

2 4 6 8 10

注意:

在函数中注册有多个回调内容时(如使用 call_user_func()call_user_func_array()),如在前一个回调中有未捕获的异常,其后的将不再被调用。