由于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,就不能实例化该对象了。