由於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,就不能例項化該物件了。