我們通常在做一個有數據庫后臺的網站的時候,都會考慮到程序需要適用于不同的應用環境。和其他編程語言有所不同的是,在PHP中,操作數據庫的是一系列的具體功能函數(如果你不使用ODBC接口的話)。這樣做雖然效率很高,但是封裝卻不夠。如果有一個統一的數據庫接口,那么我們就可以不對程序做任何修改而適用于多種數據庫,從而使程序的移植性和跨平臺能力都大大提高。
下面我們 就帶大家全面了解php面向對象的封裝、繼承、多態三大基本特征。
可以觀看我們 的相關視頻課程:第10章 PHP中的類與對象精講:這次一定要你懂
也可以??磳W知手冊:類和對象的教程
PHP面向對象的三大特征: 繼承,封裝,多態
所謂封裝,也就是把客觀事物封裝成抽象的類,并且類可以把自己的數據和方法只讓可信的類或者對象操作,對不可信的進行信息隱藏。
封裝是面向對象的特征之一,是對象和類概念的主要特性。 簡單的說,一個類就是一個封裝了數據以及操作這些數據的代碼的邏輯實體。在一個對象內部,某些代碼或某些數據可以是私有的,不能被外界訪問。通過這種方式,對象對內部數據提供了不同級別的保護,以防止程序中無關的部分意外的改變或錯誤的使用了對象的私有部分。
所謂繼承是指可以讓某個類型的對象獲得另一個類型的對象的屬性的方法,它支持按級分類的概念。
繼承是指這樣一種能力:它可以使用現有類的所有功能,并在無需重新編寫原來的類的情況下對這些功能進行擴展。 通過繼承創建的新類稱為“子類”或“派生類”,被繼承的類稱為“基類”、“父類”或“超類”。繼承的過程,就是從一般到特殊的過程。要實現繼承,可以通過“繼承”(Inheritance)和“組合”(Composition)來實現。繼承概念的實現方式有二類:實現繼承與接口繼承。實現繼承是指直接使用基類的屬性和方法而無需額外編碼的能力;接口繼承是指僅使用屬性和方法的名稱、但是子類必須提供實現的能力;
所謂多態就是指一個類實例的相同方法在不同情形有不同表現形式。
多態機制使具有不同內部結構的對象可以共享相同的外部接口。這意味著,雖然針對不同對象的具體操作不同,但通過一個公共的類,它們(那些操作)可以通過相同的方式予以調用。
一.繼承
1、如何實現繼承?
給子類使用extends關鍵字,讓子類繼承父類;
class Student extends Person{}
2、實現繼承的注意事項?
① 子類只能繼承父類的非私有屬性。
②子類繼承父類后,相當于將父類的屬性和方法copy到子類,可以直接使用$this調用。
③ PHP只能單繼承,不支持一個類繼承多個類。但是一個類進行多層繼承;
class Person{}class Chengnian extends Person{}class Student extends Chengnian{}//Student類就同時具有了Chengnian類和Person類的屬性和方法
3、方法覆蓋(方法重寫)
條件① 子類繼承父類。
條件② 子類重寫父類已有方法。
符合上述兩個條件,稱為方法覆蓋。覆蓋之后,子類調用方法,將調用子類自己的方法。
同樣,除了方法覆蓋,子類也可以具有與父類同名的屬性,進行屬性覆蓋。
4、如果,子類重寫了父類方法,如何在子類中調用父類同名方法?
partent::方法名();
所以,當子類繼承父類時,需在子類的構造中的第一步,首先調用父類構造進行復制。
function construct($name,$sex,$school){ parent::construct($name,$sex); $this->school = $school;}
實例一枚:
class Student extends Person{ //子類繼承父類 public $school; function construct($name,$sex,$school){ //子類的構造函數 parent::construct($name,$sex); //調用父類構造進行復制 $this->school = $school; } function program(){ echo "PHP真好玩!我愛PHP!PHP是世界上最好用的編程語言!<br>"; } function say(){ parent::say(); //重寫父類的同名方法 echo "我是{$this->school}的"; } } $zhangsan = new Student("張三","男","起航"); $zhangsan->say(); $zhangsan->program();
二、封裝
1、什么是封裝?
通過訪問修飾符,將類中不需要外部訪問的屬性和方法進行私有化處理,以實現訪問控制。
【注意】:是實現訪問控制,而不是拒絕訪問。 也就是說,我們私有化屬性之后,需要提供對應的方法,讓用戶通過我們提供的方法處理屬性。
2、封裝的作用?
①使用者只關心類能夠提供的功能,而不必關心功能實現的細節!(封裝方法)
②對用戶的數據進行控制,防止設置不合法數據,控制返回給用戶的數據(屬性封裝+set/get方法)
3、實現封裝操作?
① 方法的封裝
對于一些只在類內部使用的方法,而不像對外部提供使用。那么,這樣的方法我們可以使用private進行私有化處理。
private function formatName(){} //這個方法僅僅能在類內部使用$this調用
function showName(){
$this -> formatName();
}
②屬性的封裝+set/get方法
為了控制屬性的設置以及讀取,可以將屬性進行私有化處理,并要求用戶通過我們提供的set/get方法進行設置
private $age;
function setAge($age){
$this->age = $age;
}
function getAge(){
return $this->age;
}
$對象 -> getAge();
$對象 -> setAge(12);
③ 屬性的封裝+魔術方法
private $age;
function get($key){
return $this->$key;
}
function set($key,$value){
$this->$key=$value;
}
$對象->age; // 訪問對象私有屬性時,自動調用get()魔術方法,并且將訪問的屬性名傳給get()方法;
$對象->age=12; // 設置對象私有屬性時,自動調用set()魔術方法,并且將設置的屬性名以及屬性值傳給set()方法;
【 注意】:在魔術方法中,可以使用分支結構,判斷$key的不同,進行不同操作。
4、關于封裝的魔術方法:
① set($key,$value):給類私有屬性賦值時自動調用,調用時給方法傳遞兩個參數:需要設置的屬性名、屬性值;
② get($key):讀取類私有屬性時自動調用,調用時給方法傳遞一個參數:需要讀取的屬性名;
③ isset($key):外部使用isset()函數檢測私有屬性時,自動調用。
>>> 類外部使用isset();檢測私有屬性,默認是檢測不到的。false
>>> 所以,我們可以使用isset();函數,在自動調用時,返回內部檢測結果。
function isset($key){ return isset($this->$key);}
當外部使用isset($對象名->私有屬性);檢測時,將自動調用上述isset()返回的結果!
④ unset($key):外部使用unset()函數刪除私有屬性時,自動調用;
function unset($key){ unset($this->$key);}
當外部使用unset($對象名->私有屬性);刪除屬性時,自動將屬性名傳給unset(),并交由這個魔術方法處理。
實例一枚
class Person{ public $name; public $age; public $sex; function construct($name, $age,$sex){ $this->name=$name; $this->setAge($age); $this->setSex($sex); } function setAge($age){ if($age>=0&&$age<=120){ return $this->age=$age; }else{ die("年齡輸入有誤?。。?quot;); } } function setSex($sex){ if($sex=="女"||$sex=="男"){ return $this->sex=$sex; }else{ die("性別輸入有誤?。?!"); } } function say(){ echo "我的名字叫{$this->name},我的年齡{$this->age},我的性別是{$this->sex}<br>"; } }class Work extends Person{private $position; function construct($name, $age,$sex,$position){ parent::construct($name, $age,$sex); $this->job=$job; $this->setPosition($position); } function setPosition($position){ $arr=['總監','董事長','程序員','清潔工']; if(in_array($position, $arr)){ return $this->position=$position; }else{ die("不存在該職位"); } } function set($key,$value){ if($key=="age"){ return parent::setAge($value); } elseif($key=="sex"){ return parent::setSex($value); } elseif($key=="position"){ return $this->setPosition($value); } return $this->$key=$value; } function say(){ parent::say(); echo "我的職位是{$this->position}"; } } $zhangsan=new Work("張三",22,"男","總監"); $zhangsan->setSex("女"); $zhangsan->setAge(30);// $zhangsan->setPosition("董事長"); $zhangsan->position="董事長"; $zhangsan->name="lisi"; $zhangsan->say();
三.多態
3.1、什么是多態?
多態實現多態的前提是實現繼承。
1.一個類被多個子類繼承,如果這個類的某個方法在多個子類中表現出不同的功能,我們稱這種行為為多態。在PHP中的方法重寫,
2.實現多態的必要途徑:
⑴子類繼承父類;
⑵重寫父類方法;
⑶父類引用指向子類對象;
class Computer{ function fangfa(InkBox $a,Paper $b){ //父類引用 echo "即將開始打印····<br>"; $a->color(); $b->sizes(); echo "打印結束···<br>"; }}class Color implements InkBox{ function color(){ echo "正在裝載彩色墨盒<br>"; echo "實現彩色墨盒<br>"; }}class White implements InkBox{ function color(){ echo "正在裝載黑白墨盒<br>"; echo "實現黑白墨盒<br>"; }}class A4 implements Paper{ function sizes(){ echo "正在加載A4紙張<br>"; echo "實現A4紙張<br>"; }}class A5 implements Paper{ function sizes(){ echo "實現A5紙張<br>"; }}$com=new Computer();//創建對象$com->fangfa(new Color(),new A4());//子類對象
相關課程推薦:
視頻教程: 韓順平 2016年 最新PHP面向對象編程視頻教程
視頻教程: PHP面向對象編程視頻教程
鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。
新聞熱點
疑難解答