今天在群里面,有個叫lewis的在問call_user_func_array的用法,因為之前一直沒有用過,也不能說什么,於是看一下手冊,發現是這么寫的:
call_user_func_array
(PHP 4 >= 4.0.4, PHP 5) call_user_func_array -- Call a user function given with an array of parameters
Description
mixed call_user_func_array ( callback function, array param_arr )
Call a user defined function given by function, with the parameters in param_arr.
然后還有一個例子:
<?php function debug($var, $val) { echo "***DEBUGGING VARIABLE: $var VALUE:"; if (is_array($val) || is_object($val) || is_resource($val)) { print_r($val); } else { echo " $val "; } echo "*** "; } $c = mysql_connect(); $host = $_SERVER["SERVER_NAME"]; call_user_func_array('debug', array("host", $host)); call_user_func_array('debug', array("c", $c)); call_user_func_array('debug', array("_POST", $_POST)); ?>
相信看了例子之后應該有點明白了吧?
我自己是這么理解這個函數的,如果說的不對,還望各位高手不要恥笑:
該函數真正的用法有點類似於函數重載,因為他的第一個參數是字符型的,也就是函數的名稱,第二個參數是數組,我們可以當成該函數的各個參數,而事實上也就是這么用的,如果你看過我的前一篇文章:PHP的偽重載 ,或許你能夠理解,正是因為這個函數的存在,我發現函數重載也可以這樣運用:
<?php /** * 例子寫完后,本來認為完事了,結果遇到有人問call_user_func_array(),看了一下手冊 * 原來,我上面的那個test函數還可以精簡成如下的例子, */ function otest1 ($a) { echo('一個參數'); } function otest2($a, $b) { echo('二個參數'); } function otest3 ($a, $b, $c) { echo('三個啦'); } function otest() { $args = func_get_args(); $num = func_num_args(); call_user_func_array('otest' . $num, $args); } otest(1, 2); ?>
看到不?而我最初的寫法,在PHP的偽重載一文中有所提及,僅作參考。。。。
這些只是call_user_func_array的簡易用法,在PHP4下測試過,而手冊中還有一些將第一個參數當成數組來傳入的例子,我在PHP4下是沒有辦法運行的,也許PHP5可以吧,但我不用PHP5的,也沒有辦法解釋什么。謝謝各位
PHP函數call_user_func和call_user_func_array詳解
call_user_func 函數類似於一種特別的調用函數的方法,使用方法如下:
1.調用 普通函數:
<?php function a($b, $c) { echo $b; echo $c; } call_user_func('a', "111", "222"); call_user_func('a', "333", "444"); //顯示 111 222 333 444 ?>
調用類內部的方法比較奇怪,居然用的是array,不知道開發者是如何考慮的,當然省去了new,也是滿有新意的:
2.調用 類的方法(包括類的靜態的方法與實例對象方法)
<?php class a { function b($i) { echo $i; } public static c($k) { echo $k; } } //當php <5.3時,可以如下使用,此時會把 b()方法當作是a的一個靜態方式。 call_user_func(array("a", "b"), "111"); //當php >=5.3時,類的公開的非靜態的方法必須在類實例化后方可被調用,否則會提示Strict性錯誤(為了兼容先前及以后的版本,還是用對象方法傳入)。 $obj = new a; call_user_func(array($obj, "b"), "111");//顯示 111 //靜態方法可以如下方式調用 call_user_func(array("a", "b"), "111"); //或 call_user_func("a::b","111"); ?>
call_user_func_array函數和call_user_func很相似,只不過是換了一種方式傳遞了參數,讓參數的結構更清晰:
<?php function a($b, $c) { echo $b; echo $c; } call_user_func_array('a', array("111", "222")); //顯示 111 222 ?>
call_user_func_array 函數也可以調用類內部的方法的,只不過是后面傳參要以數組的形式傳入而已。
<?php Class ClassA { function bc($b, $c) { $bc = $b + $c; echo $bc; } function d() { $bc = $b + $c; echo $bc; } } //php<5.3時,非靜態的方法可直接傳入類名 call_user_func_array(array('ClassA', 'bc'), array("111", "222")); //php>=5.3時,非靜態的方法 只有在類被實例化后方可調用,否則會提示Strict性錯誤 $obj = new classA; call_user_func_array(array($obj, 'bc'), array("111", "222")); //靜態方法調用如下 call_user_func_array(array('ClassA','bc'), array("111", "222")); //或 call_user_func_array('ClassA::bc', array("111", "222")); ?>
call_user_func函數和call_user_func_array函數都支持引用,這讓他們和普通的函數調用更趨於功能一致:
<?php function a(&$b) { $b++; } $c = 0; call_user_func('a', &$c);//注意,5.*版本中,call_user_func不提倡引用傳遞,提示已過時。 echo $c;//顯示 1 call_user_func_array('a', array(&$c)); echo $c;//顯示 2 ?>
官方手冊內容如下: