PHP数据类型之Callback / Callable 类型

@爱耍流氓的唐僧  July 2, 2020

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

传值:
PHP是将函数以string形式传递的。可以使用任何内置函数和自定义函数,不能使用语言结构(array(),echo,empty(),eval(),exit(),isset(),list(),print 或 unset())。

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

静态类方法也可不经实例化该类的对象而传递,只要在下标 0 中包含类名而不是对象。

除了普通的用户自定义函数外,也可传递 匿名函数 给回调参数。
Example1:

    <?php 

// An example callback function
function my_callback_function() {
    echo 'hello world!';
}

// An example callback method
class MyClass {
    static function myCallbackMethod() {
        echo 'Hello World!';
    }
}

// Type 1: Simple callback
call_user_func('my_callback_function'); 

// Type 2: Static class method call
call_user_func(array('MyClass', 'myCallbackMethod')); 

// Type 3: Object method call
$obj = new MyClass();
call_user_func(array($obj, 'myCallbackMethod'));

// Type 4: Static class method call (As of PHP 5.2.3)
call_user_func('MyClass::myCallbackMethod');

// Type 5: Relative static class method call (As of PHP 5.3.0)
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

// Type 6: Objects implementing __invoke can be used as callables (since PHP 5.3)
class C {
    public function __invoke($name) {
        echo 'Hello ', $name, "\n";
    }
}

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

Example2

<?php
// Our closure
$double = function($a) {
    return $a * 2;
};

// This is our range of numbers
$numbers = range(1, 5);

// Use the closure as a callback here to 
// double the size of each element in our 
// range
$new_numbers = array_map($double, $numbers);

print implode(' ', $new_numbers);
?>

以上例程会输出:

2 4 6 8 10

如果一个方法需要接受一个回调方法作为参数,我们可以这样写

<?php
function dosth($callback){
    call_user_func($callback);
}

function callback(){
    echo 'do sth callback';
}

dosth('callback');

但我们不能确定回调方法是否可以调用,因此需要做很多额外的工作去检查这个回调方法是否可以调用。
有什么更好的方法可以判断回调方法是否可调用?
我们可以使用callable来强制指定参数为回调类型,这样可以保证回调方法必须是可以调用的。

<?php
function dosth(callable $callback){
    call_user_func($callback);
}

dosth('abc');

执行后,提示错误:TypeError: Argument 1 passed to dosth() must be callable ,程序并不能执行到dosth内部的处理,从参数类型处就已经做了检查处理,起到保护作用。
而如果把callable去掉,程序就能执行到dosth内部的处理,这样就需要做很多额外的工作去检查这个回调方法是否可以调用。因此,如果方法的参数是回调方法,都应加上callable强制指定为回调类型,这样可以减少调用的错误和提高程序的质量。


添加新评论