多個函數用同一個名字,但參數表,即參數的個數或(和)數據類型可以不同,調用的時候,雖然方法名字相同,但根據參數表可以自動調用對應的函數。
PHP4 中僅僅實現了html' target='_blank'>面向對象的部分的、簡單的功能,而 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 'method2'; } } //通過在類中的額外的處理,使用這個類對用戶是透明的: $obj1 = new Myclass('A'); //將調用 method1 $obj2 = new Myclass('B','C'); //將調用 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 == 'display') { 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('Cat'); ?>
以上代碼中,調用了 display() 方法,可以根據參數的類型和個數調用類中的對應的代碼段,從而實現了對象方法的重載。
以上就是實例比較php方法重載的兩種方式的詳細內容,更多請關注 其它相關文章!
鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。
新聞熱點
疑難解答