首頁 >後端開發 >php教程 >php中兩種重載版本比較區別詳解

php中兩種重載版本比較區別詳解

伊谢尔伦
伊谢尔伦原創
2017-06-30 10:23:351319瀏覽

「重載」是類別的多態的一種實作。函數重載指一個標識符被用作多個函數名,並且能夠透過函數的參數個數或參數型別將這​​些同名的函數區分開來,呼叫不發生混淆。這樣做的主要好處就是,不用為了對不同的參數型態或參數個數,而寫多個函數。

多個函數用同一個名字,但參數表,即參數的個數或(和)資料型別可以不同,呼叫的時候,雖然方法名字相同,但根據參數表可以自動呼叫對應的函數。

PHP4 中僅僅實現了物件導向的部分的、簡單的功能,而 PHP5 以後對物件的支援就強大的多了。

對於多態的實現,PHP4 只支援覆蓋(override),而不支援重載(overload)。但我們可以透過一些技巧來「模擬」重載的實現。

PHP5 雖然可以支援覆蓋和重載,但重載在具體實作上,和其他語言還有較大的差異。

1,在PHP4 中「模擬」重載

試看以下程式碼:

<?php 
//根据参数个数选择执行不同的方法(在 PHP4 中模拟"重载"(多态的一种) 
class Myclass 
{ 
function Myclass() 
{ 
$method = "method" . func_num_args(); 
$this->$method(); 
} 
function method1($x) 
{ 
echo "method1"; 
} 
function method2($x, $y) 
{ 
echo &#39;method2&#39;; 
} 
} 
//通过在类中的额外的处理,使用这个类对用户是透明的: 
$obj1 = new Myclass(&#39;A&#39;); //将调用 method1 
$obj2 = new Myclass(&#39;B&#39;,&#39;C&#39;); //将调用 method2 
?>

以上程式碼中,透過在建構子中使用func_num_args() 函數取到參數的個數,自動執行method1 或method2 方法。我們可以結合函數 func_get_arg(i) 和 func_get_args() 來改進上述範例。

2,在PHP5 中使用重載

先看以下範例:

<?php 
class Myclass 
{ 
public $attriable; 
public $one = "this is one"; 
public $two = "this is two"; 
function construct() 
{ 
} 
function one($one) 
{ 
$this->one=$one; 
$this->attriable = $this->one; 
} 
function one($one, $two) 
{ 
$this->one=$one; 
$this->two=$two; 
$this->attriable = $this->one . $this->two; 
} 
function display() 
{ 
echo $this->attriable; 
} 
} 
$one = "this is my class"; 
$two = "Im the best"; 
$myclass = new myclass(); 
$myclass->one($one); 
$myclass->display(); 
$myclass->one($one, $two); 
$myclass->display(); 
//本例的做法,在 PHP 中是不正确的! 
?>


使用過C++、Java、C# 重載的人,很習慣就會寫出以上的重載實作的PHP 程式碼。但這在 PHP5 中是不正確的。 PHP5 並不是對前述幾種語言的模仿,而是有自己的一套實作方法重載的方法(是好是壞,這裡不討論)。 雖然 PHP5 的類別較 PHP4 強大了許多,但在「重載」這個問題上並沒有像我們預期的那樣「改善」。在“強”類型的語言中可以透過不同的參數類型來實現“重載”,例如C++、Java、C# 等。在「固定參數」傳遞的語言中,也可以透過參數的數量傳遞,例如 Java,但是 PHP 是弱型別語言,因此不會出現類似以上的「重載」。

PHP5 中重載可以透過 get, set, and call 幾個特殊方法來進行。當 Zend 引擎試圖存取一個成員並沒有找到時,PHP將會呼叫這些方法。

在以下範例中,get和set取代所有對屬性變數陣列的存取。如果必要,你還可以實現任何類型你想要的過濾。例如,腳本可以禁止設定屬性值, 在開始時用一定的前綴或包含一定類型的值。 call 方法說明了你如何呼叫未經定義的方法。你呼叫未定義方法時,方法名稱和方法接收的參數將會傳給call方法, PHP傳遞call的值傳回給未定義的方法。

<?php 
class Overloader 
{ 
private $properties = array(); 
function get($property_name) 
{ 
if(isset($this->properties[$property_name])) 
{ 
return($this->properties[$property_name]); 
} 
else 
{ 
return(NULL); 
} 
} 
function set($property_name, $value) 
{ 
$this->properties[$property_name] = $value; 
} 
public function call($method, $p) 
{ 
print("Invoking $method()<br>\n"); 
//print("Arguments: "); 
//print_r($args); 
if($method == &#39;display&#39;) 
{ 
if(is_object($p[0])) 
$this->displayObject($p[0]); 
else 
if(is_array($p[0])) 
$this->displayArray($p[0]); 
else 
$this->displayScalar($p[0]); 
} 
} 
public function displayObject($p) 
{ 
echo ("你传入的是个对象,内容如下:<br>"); 
print_r($p); 
echo "<hr>"; 
} 
public function displayArray($p) 
{ 
echo ("你传入的是个数组,内容如下:<br>"); 
print_r($p); 
echo "<hr>"; 
} 
public function displayScalar($p) 
{ 
echo ("你传入的是个单独变量,内容如下:<br>" . $p); 
echo "<hr>"; 
} 
} 
$o = new Overloader(); 
//调用 set() 给一个不存在的属性变量赋值 
$o->dynaProp = "Dynamic Content"; 
//调用 get() 
print($o->dynaProp . "<br>\n"); 
//调用 call() 
//$o->dynaMethod("Leon", "Zeev"); 
$o->display(array(1,2,3)); 
$o->display(&#39;Cat&#39;); 
?>


以上程式碼中,呼叫了display() 方法,可以根據參數的類型和個數呼叫類別中的對應的程式碼段,從而實現了物件方法的重載。

以上是php中兩種重載版本比較區別詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn