由於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,就不能實例化該對象了。