在PHP中,回调(Callback)可以通过 callable 类型声明来表示。一些函数如 call_user_func() 或 usort() 可以接受用户自定义的回调函数作为参数。回调函数不止可以是简单函数,还可以是对象的方法,包括静态类方法。
传递
- PHP是将函数以string形式传递的。 可以使用任何内置或用户自定义函数,但除了语言结构例如:array(),echo,empty(),eval(),exit(),isset(),list(),print 或 unset()。
- 一个已实例化的 object 的方法被作为 array 传递,下标 0 包含该 object,下标 1 包含方法名。 在同一个类里可以访问 protected 和 private 方法。
- 静态类方法可以不实例化 object 传递,只需要在下标为 0 的位置传递类名而不是 object ,或者传递 ‘ClassName::methodName’。
- 回调参数不仅可以使用普通的用户自定义函数,也接受 匿名函数 和 箭头函数。
注意:从 PHP 8.1.0 开始,还可以使用 First-class 可调用语法 创建匿名函数。
通常情况下,任何实现了 __invoke() 的对象都可以传入回调参数。
示例:回调函数示例
<?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 A { public static function who() { echo "A\n"; } } class B extends A { public static function who() { echo "B\n"; } } call_user_func(array('B', 'parent::who')); // A,从 PHP 8.2.0 起弃用。 // 类型 6:实现 __invoke 的对象用于回调 class C { public function __invoke($name) { echo 'Hello ', $name, "\n"; } } $c = new C(); call_user_func($c, 'PHP!'); ?>
示例 :使用 Closure 的示例
<?php // 闭包 $double = function($a) { return $a * 2; }; // 这是数字范围 $numbers = range(1, 5); // 这里使用闭包作为回调, // 将范围内的每个元素数值翻倍 $new_numbers = array_map($double, $numbers); print implode(' ', $new_numbers); ?>
以上示例会输出:
2 4 6 8 10
注意:在函数中注册有多个回调内容时(如使用 call_user_func() 与 call_user_func_array()),如在前一个回调中有未捕获的异常,其后的将不再被调用。