PHP函数call_user_func和call_user_func_array详解

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
?>

官方手册内容如下:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值