PHP构造函数的重载
由于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,就不能实例化该对象了。