PHP中實現函數重載

因爲PHP是弱類型語言,所以函數的輸入參數類型沒法肯定(可使用類型暗示,可是類型暗示沒法用在諸如整型,字符串之類的標量類型上),而且對於一個函數,好比只定義了3個輸入參數,PHP卻運行調用的時候輸入4個或者更多的參數。所以基於這2點,註定了PHP中沒法重載函數,(相似Javascript語言),也沒法有構造函數的重載。php

因爲實現函數的重載對提升開發效率頗有幫助,若是能象C#或者C++那樣,那就很是好了。事實上,PHP的提供了一個魔術方法,mixed __call ( string name, array arguments )。這個方法在php手冊中也有說起,根據官方文檔,稱此方法能夠實現函數重載。當調用對象中一個不存在的方法的時候,若是定義了__call()方法,則會調用該方法。好比下面的代碼:c#

<?php
class A
{
   function __call ( $name, $arguments )
    {
        echo "__call調用<br/>";
        echo '$name爲'.$name."<br/>";
        print_r ($arguments);
    }
}
(new A)->test("test","argument");
?>

運行結果爲:ide

__call調用函數

$name爲testthis

Array ( [0] => test [1] => argument ) 對象

所以只須要利用該魔術方法既能夠實現函數重載。ip

代碼以下:開發

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

相關文章
相關標籤/搜索