由於PHP是弱型別語言,因此函式的輸入引數型別無法確定(可以使用型別暗示,但是型別暗示無法用在諸如整型,字串之類的標量型別上),並且對於一個函式,比如只定義了3個輸入引數,PHP卻執行呼叫的時候輸入4個或者更多的引數。因此基於這2點,註定了PHP中無法過載函式,(類似Javascript語言),也無法有建構函式的過載。
由於實現函式的過載對提高開發效率很有幫助,如果能象C#或者C++那樣,那就非常好了。事實上,PHP的提供了一個魔術方法,mixed __call ( string name, array arguments )。這個方法在php手冊中也有提及,根據官方文件,稱此方法可以實現函式過載。當呼叫物件中一個不存在的方法的時候,如果定義了__call()方法,則會呼叫該方法。比如下面的程式碼:
<?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 )
因此只需要利用該魔術方法既可以實現函式過載。
程式碼如下:
<?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中,實現建構函式的過載,也只能變通的實現。實現的關鍵要素是獲取輸入引數,並且根據輸入引數判斷呼叫哪個方法。下面是一個示例程式碼:
<?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,就不能例項化該物件了。