PHP函数call_user_func和call_user_func_array详解
今天在群里面,有个叫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 ?>
官方手册内容如下: