在學習PHP 這種語言中你會發現, PHP中的方法是不能重載的, 所謂的方法重載就是定義相同的方法名,通過“參數的個數“不同或“參數的類型“不 同,來訪問我們的相同方法名的不同方法。但是因為PHP是弱類型的語言, 所以在方法的參數中本身就可以接收不同類型的數據,又因為PHP的方法可以接收不定個數的參數,所以通過傳遞不同個數的參數調用不相同方法名的不同方法也是不成立的。所以在PHP里面沒有方法重載。不能重載也就是在你的項目中不能定義相同方法名的方法。另外,因為PHP沒有名子空間的概念,在同一個頁面和被包含的頁面中不能定義相同名稱的方法, 也不能定義和PHP給我提供的方法的方法重名,當然在同一個類中也不能定義相同名稱的方法。
我們這里所指的重載新的方法所指的是什么呢?其實我們所說的重載新的方法就是子類覆蓋父類的已有的方法,那為什么要這么做呢?父類的方法不是可以繼承過 來直接用嗎?但有一些情況是我們必須要覆蓋的,比如說我們前面提到過的例子里面, “Person”這個人類里面有一個“說話”的方法,所有繼承“Person”類的子類都是可以“說話”的, 我們“Student”類就是“Person”類的子類,所以“Student”的實例就可以“說話“了, 但是人類里面“說話”的方法里面說出的是“Person”類里面的屬性, 而“Student”類對“Person”類進行了擴展,又擴展出了幾個新的屬性,如果使用繼承過來的“say()”說話方法的話,只能說出從 “Person”類繼承過來的那些屬性,那么新擴展的那些屬性使用這個繼承過來的“say()”的方法就說不出來了,那有的人就問了,我在 “Student”這個子類中再定義一個新的方法用於說話,說出子類里面所有的屬性不就行了嗎?一定不要這么做, 從抽象的角度來講, 一個“學生”不能有兩種“說話”的方法,就算你定義了兩個不同的說話的方法,可以實現你想要的功能,被繼承過來的那個“說話“方法可能沒有機會用到了,而 且是繼承過來的你也刪不掉。這個時候我們就要用到覆蓋了。
雖然說在PHP里面不能定義同名的方法, 但是在父子關系的兩個類中,我們可以在子類中定義和父類同名的方法,這樣就把父類中繼承過來的方法覆蓋掉了。
<? // 定義一個"人"類做為父類 class Person { // 下面是人的成員屬性 var $name; // 人的名子 var $sex; // 人的性別 var $age; // 人的年齡 // 定義一個構造方法參數為屬性姓名$name、性別$sex和年齡$age進行賦值 function __construct($name, $sex, $age) { $this->name = $name; $this->sex = $sex; $this->age = $age; } // 這個人可以說話的方法, 說出自己的屬性 function say() { echo "我的名子叫:" . $this->name . " 性別:" . $this->sex . " 我的年齡是:" . $this->age; } } class Student extends Person { var $school; // 學生所在學校的屬性 // 這個學生學習的方法 function study() { echo "我的名子叫:" . $this->name . " 我正在" . $this->school . " 學習"; } // 這個學性可以說話的方法, 說出自己所有的屬性,覆蓋了父類的同名方法 function say() { echo "我的名子叫:" . $this->name . " 性別:" . $this->sex . " 我的年齡是:" . $this->age . " 我在" . $this->school . "上學"; } } ?>
上面的例子, 我們就在“Student”子類里覆蓋了繼承父類里面的”say()”的方法,通過覆蓋我們就實現了對“方法”擴展。但是,像這樣 做雖然解決了我們上面說的問題,但是在實際開發中,一個方法不可能就一條代碼或是幾條代碼,比如說“Person”類里面的“say()”方法有里面有 100條代碼,如果我們想對這個方法覆蓋保留原有的功能外加上一點點功能,就要把原有的100條代碼重寫一次, 再加上擴展的幾條代碼,這還算是好的,而有的情況,父類中的方法是看不見原代碼的,這個時候你怎么去重寫原有的代碼呢?我們也有解決的辦法,就是在子類這 個方法中可以調用到父類中被覆蓋的方法, 也就是把被覆蓋的方法原有的功能拿過來再加上自己的一點功能,可以通過兩種方法實現在子類的方法中調用父類被覆蓋的方法:
一種是使用父類的“類名::“來調用父類中被覆蓋的方法;
一種是使用“parent::”的方試來調用父類中被覆蓋的方法;
class Student extends Person { var $school; // 學生所在學校的屬性 // 這個學生學習的方法 function study() { echo "我的名子叫:" . $this->name . " 我正在" . $this->school . "學習"; } // 這個學性可以說話的方法, 說出自己所有的屬性,覆蓋了父類的同名方法 function say() { // 使用父類的"類名::"來調用父類中被覆蓋的方法; // Person::say(); // 或者使用"parent::"的方試來調用父類中被覆蓋的方法; parent::say(); // 加上一點自己的功能 echo "我的年齡是:" . $this->age . " 我在" . $this->school . "上學"; } }
現在用兩種方式都可以訪問到父類中被覆蓋的方法,我們選那種方式最好呢?用戶可能會發現自己寫的代碼訪問了父類的變量和函數。如果子類非常精煉或者父類非 常專業化的時候尤其是這樣。 不要用代碼中父類文字上的名字,應該用特殊的名字 parent,它指的就是子類在 extends 聲明中所指的父類的名字。這樣做可以避免在多個地方使用父類的名字。如果繼承樹在實現的過程中要修改,只要簡單地修改類中 extends 聲明的部分。
同樣,構造方法在子類中如果沒有聲明的話,也可以使用父類中的構造方法,如果子類中重新定義了一個構造方法也會覆蓋掉父類中的構造方法,如果想使用新的構造方法為所有屬性賦值也可以用同樣的方式。
class Student extends Person { var $school; // 學生所在學校的屬性 function __construct($name, $sex, $age, $school) { // 使用父類中的方法為原有的屬性賦值 parent::__construct($name, $sex, $age); $this->school = $school; } // 這個學生學習的方法 function study() { echo "我的名子叫:" . $this->name . " 我正在" . $this->school . " 學習"; } // 這個人可以說話的方法, 說出自己的屬性 function say() { parent::say(); // 加上一點自己的功能 echo "我的年齡是:" . $this->age . " 我在" . $this->school . "上學"; } }