Callback / Callable 類型

回調(diào)可以通過 callable 類型聲明來表示。

一些函數(shù)如 call_user_func()usort() 可以接受用戶自定義的回調(diào)函數(shù)作為參數(shù)。回調(diào)函數(shù)不止可以是簡單函數(shù),還可以是對象的方法,包括靜態(tài)類方法。

傳遞

PHP是將函數(shù)以string形式傳遞的。 可以使用任何內(nèi)置或用戶自定義函數(shù),但除了語言結(jié)構(gòu)例如:array()echo,empty()eval(),exit(),isset(),list()printunset()。

一個已實例化的 object 的方法被作為 array 傳遞,下標(biāo) 0 包含該 object,下標(biāo) 1 包含方法名。 在同一個類里可以訪問 protected 和 private 方法。

靜態(tài)類方法可以不實例化 object 傳遞,只需要在下標(biāo)為 0 的位置傳遞類名而不是 object ,或者傳遞 'ClassName::methodName'。

回調(diào)參數(shù)不僅可以使用普通的用戶自定義函數(shù),也接受 匿名函數(shù)箭頭函數(shù)。

注意:

從 PHP 8.1.0 開始,還可以使用 First-class 可調(diào)用語法 創(chuàng)建匿名函數(shù)。

通常情況下,任何實現(xiàn)了 __invoke() 的對象都可以傳入回調(diào)參數(shù)。

示例 #1 回調(diào)函數(shù)示例

<?php 

// 回調(diào)函數(shù)示范
function my_callback_function() {
    echo 
'hello world!';
}

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

// 類型 1:簡單的回調(diào)
call_user_func('my_callback_function'); 

// 類型 2:靜態(tài)類方法回調(diào)
call_user_func(array('MyClass''myCallbackMethod')); 

// 類型 3:對象方法回調(diào)
$obj = new MyClass();
call_user_func(array($obj'myCallbackMethod'));

// 類型 4:靜態(tài)類方法回調(diào)
call_user_func('MyClass::myCallbackMethod');

// 類型 5:父級靜態(tài)類回調(diào)
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:實現(xiàn) __invoke 的對象用于回調(diào)
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;
};

// 這是數(shù)字范圍
$numbers range(15);

// 這里使用閉包作為回調(diào),
// 將范圍內(nèi)的每個元素數(shù)值翻倍
$new_numbers array_map($double$numbers);

print 
implode(' '$new_numbers);
?>

以上例程會輸出:

2 4 6 8 10

注意:

在函數(shù)中注冊有多個回調(diào)內(nèi)容時(如使用 call_user_func()call_user_func_array()),如在前一個回調(diào)中有未捕獲的異常,其后的將不再被調(diào)用。