由于PHP是弱类型语言,因此函数的输入参数类型无法确定(可以使用类型暗示,但是类型暗示无法用在诸如整型,字符串之类的标量类型上),并且对于一个函数,比如只定义了3个输入参数,PHP却运行调用的时候输入4个或者更多的参数。因此基于这2点,注定了PHP中无法重载函数,(类似Javascript语言),也无法有构造函数的重载。
由于实现函数的重载对提高开发效率很有帮助,如果能象C#或者C++那样,那就非常好了。事实上,PHP的提供了一个魔术方法,mixed __call ( string name, array arguments )。这个方法在php手册中也有提及,根据官方文档,称此方法可以实现函数重载。当调用对象中一个不存在的方法的时候,如果定义了__call()方法,则会调用该方法。比如下面的代码:
1
2
3
4
5
6
7
8
9
10
11
12
|
<?php
class A
{
function __call ( $name , $arguments )
{
echo "__call调用<br/>" ;
echo '$name为' . $name . "<br/>" ;
print_r ( $arguments );
}
}
( new A)->test( "test" , "argument" );
?>
|
运行结果为:
__call调用
$name为test
Array ( [0] => test [1] => argument )
因此只需要利用该魔术方法既可以实现函数重载。
代码如下:
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
28
29
30
|
<?php
class A
{
function __call ( $name , $args )
{
if ( $name == 'f' )
{
$i = count ( $args );
if (method_exists( $this , $f = 'f' . $i )) {
call_user_func_array( array ( $this , $f ), $args );
}
}
}
function f1( $a1 )
{
echo "1个参数" . $a1 . "<br/>" ;
}
function f2( $a1 , $a2 )
{
echo "2个参数" . $a1 . "," . $a2 . "<br/>" ;
}
function f3( $a1 , $a2 , $a3 )
{
echo "3个参数" . $a1 . "," . $a2 . "," . $a3 . "<br/>" ;
}
}
( new A)->f( 'a' );
( new A)->f( 'a' , 'b' );
( new A)->f( 'a' , 'b' , 'c' );
?>
|
输出如下:
1个参数a 2个参数a,b 3个参数a,b,c
同样的在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
26
27
28
29
|
<?php
class A
{
function __construct()
{
echo "执行构造函数<br/>" ;
$a = func_get_args();
$i = count ( $a );
if (method_exists( $this , $f = '__construct' . $i )) {
call_user_func_array( array ( $this , $f ), $a );
}
}
function __construct1( $a1 )
{
echo "1个参数" . $a1 . "<br/>" ;
}
function __construct2( $a1 , $a2 )
{
echo "2个参数" . $a1 . "," . $a2 . "<br/>" ;
}
function __construct3( $a1 , $a2 , $a3 )
{
echo "3个参数" . $a1 . "," . $a2 . "," . $a3 . "<br/>" ;
}
}
$o = new A( 'a' );
$o = new A( 'a' , 'b' );
$o = new A( 'a' , 'b' , 'c' );
?>
|
执行构造函数 1个参数a 执行构造函数 2个参数a,b 执行构造函数 3个参数a,b,c
顺便提一下,和c#等面向对象语言一样,php中,把构造函数设成私有或者protected,就不能实例化该对象了。
|
请发表评论