PHP函数call_user_func和call_user_func_array详解
52PHP
革命尚未成功,同志仍须努力
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.
然后还有一个例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
<?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的伪重载 ,或许你能够理解,正是因为这个函数的存在,我发现函数重载也可以这样运用:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
<?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.调用 普通函数:
1
2
3
4
5
6
7
8
9
|
<?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.调用 类的方法(包括类的静态的方法与实例对象方法)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
<?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很相似,只不过是换了一种方式传递了参数,让参数的结构更清晰:
1
2
3
4
5
6
7
8
9
10
|
<?php
function a( $b , $c )
{
echo $b ;
echo $c ;
}
call_user_func_array( 'a' , array ( "111" , "222" ));
//显示
111 222
?>
|
call_user_func_array 函数也可以调用类内部的方法的,只不过是后面传参要以数组的形式传入而已。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
|
<?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函数都支持引用,这让他们和普通的函数调用更趋于功能一致:
1
2
3
4
5
6
7
8
9
10
11
|
<?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
?>
|
官方手册内容如下:
相关推荐
- [PHP源码阅读]explode和implode函数
- php关于变量的几个函数和常量
- Excel函数"SUMIFS"和"COUNTIFS"详解
- kcf跟踪算法代码运行流程以及代码结构和函数详解(c++)
- 详解Python中函数和模块的特殊属性__annotations__
- 详解Python中函数和模块的特殊属性__annotations__
- 好程序员分享JavaScript事件委托代理和函数封装详解
- php中关于img2thumb函数的图片裁剪规则详解
- PHP扩展参数定义结构和操作详解
- 基于PHP和JS的AES相互加密解密方法详解
- 【学习笔记】编译Linux内核(下)---KConfig、Makefile详解以及ARM平台Linux内核的编译
- Linux 调度器模拟