以下是文章本文:
一、封裝性
封裝就是把抽取出來的資料和對資料的操作封裝在一起,資料被保護在內部,程式的其他部分只有被授權的操作(方法)才能對資料進行操作。
php提供了三種存取控制修飾符
public 表示全域,本類內部,類外部,子類都可以訪問
protected 表示受保護的,只有本類或子類可以訪問
private 表示私人的,只有本類內部可以訪問
以上三種修飾符既可以修飾方法也可以修飾屬性(變數),方法如果沒有存取修飾詞則預設是public,成員屬性必須指定存取修飾詞,在PHP4中也有這種寫法 var $name,表示公開屬性,不推薦這種寫法
例:
<?php class Person{ public $name; protected $age; private $salary; function __construct($name,$age,$salary){ $this->name=$name; $this->age=$age; $this->salary=$salary; } public function showinfo(){ //這表示三個修飾符都可以在本類內部使用 echo $this->name."||".$this->age."||".$this->salary; } } $p1=new Person('張三',20,3000); //這裡屬於類外部,那麼如果用下面的方法訪問age和salary都會報錯 // echo $p1->age; echo$p1->salary; ?>
那麼現在就想在外部存取protected和private的元素和方法該怎麼辦?
通常做法是通過public函數去訪問這些變數 格式:
public function setxxxx($val){ $this->xxxx=$val; } public function getxxxx(){ return $this->xxxx; }
這裡帶set和get只是為了識別方便,並非必要
如:
public function getsalary(){ return $this->salary; //擴充:這裡可以調用一些方法,如判斷使用者名稱等,正確才給訪問 }
在外部就可以通過 echo $p1->getsalary();
如果要訪問 protected和private也可以使用以下方法,但不推薦使用,只要瞭解即可
__set() 和 __get()
__set()對protected或private屬性進行賦值操作
__set($name,$val);
__get()擷取 protected 或 private的值
__get($name);
如:
<?php class testa{ protected $name; //使用__set()來管理所有屬性 public function __set($pro_name,$pro_val){ //上面$pro_name和$pro_val可自訂 //下面$this->pro_name為既定,不可更改 $this->pro_name=$pro_val; } //使用__get()來擷取所有屬性值 public function __get($pro_name){ if(isset($pro_name)){ return $this->pro_name; } else { return null; } } } $n1=new testa(); //正常情況,類外部是不能訪問protected屬性的,但是用了上面的方法就可以對它們進行操作 $n1->name='小三'; echo $n1->name; ?>
二、繼承性
先看一個例子:
<?php class Pupil{ public $name; protected $age; public function getinfo(){ echo $this->name.'||'.$this->age; } public function testing(){ echo 'this is pupil'; } } class Graduate{ public $name; protected $age; public function getinfo(){ echo $this->name.'||'.$this->age; } public function testing(){ echo 'this is Graduate'; } } ?>
從上面的例子可以看出,當多個類有很多共同屬性和方法時,
代碼的複用性不高,代碼冗餘,思考css中的處理方法
解決方案 :繼承
<?php class Students{ public $name; public $age; public function __construct($name,$age){ $this->name=$name; $this->age=$age; } public function showinfo(){ echo $this->name.'||'.$this->age; } } class Pupil extends Students{ function testing(){ echo 'Pupil '.$this->name.' is testing'; } } class Graduate extends Students{ function testing(){ echo 'Graduate '.$this->name.' is testing'; } } $stu1=new Pupil('張三',20); $stu1->showinfo(); echo '<br/>'; $stu1->testing(); ?>
從上面可以看出,繼承就是一個子類(Subclass)通過 extends 父類
把父類(BaseClass)中的public 和 protected 的屬性和方法繼續下來,
不能繼承private屬性和方法
文法結構:
class 父類名{} class 子類名 extends 父類名{}
細節:
1、一個子類只能繼承一個父類(這裡指直接繼承);如果希望繼承多個類的屬性和方法,
可以使用多層繼承
例:
<?php class A{ public $name='AAA'; } class B extends A{ public $age=30; } class C extends B{} $p=new C(); echo $p->name;//這裡會輸出AAA ?>
2、在建立某個子類對象時,預設情況下不會自動調用其父類的建構函式
例:
class A{ public function __construct(){ echo 'A'; } } class B extends A{ public function __construct(){ echo 'B'; } } $b=new B();//這裡會優先輸出B中的構造方法,
如果B中沒有構造方法才會輸出A中的
3、在子類中如果需要訪問父類的方法(構造方法、成員方法 方法的修飾符為protected或private),
那麼可以使用 父類::方法名 或者 parent::方法名 來完成
這裡parent和以前提到的self都均為小寫,大寫報錯】
class A{ public function test(){ echo 'a_test'; } } class B extends A{ public function __construct(){ //兩種方法都行 A::test(); parent::test(); } } $b=new B();
5、如果一個子類(衍生類別)的方法與父類的方法完全一樣時(public,protected),
我們稱為方法覆蓋或方法重寫(override),看下面的多態性
三、多態性
例 :
<?php class Animal{ public $name; public $price; function cry(){ echo 'i don\'t know'; } } class Dog extends Animal{ //覆蓋、重寫 function cry(){ echo 'Wang Wang!'; Animal::cry();//這裡不會報錯,能正確執行父類的cry(); } } $dog1=new Dog(); $dog1->cry(); ?>
小結:
1、當一個父類知道所有的子類都有一個方法,但是父類不能確定該方法如何寫,可以讓子類去覆蓋它的方法,方法覆蓋(重寫),必須要求子類的方法名和參數個數完全一致
2、如果子類要去調用父類的某個方法(protected/public),可以使用 父類名::方法名 或者 parent::方法名
3、在實現方法重寫的時候,存取修飾詞可以不一樣,但是子類方法的存取權限必須大於等於父類方法的存取權限(即不能縮小父類方法的存取權限)
如 父類public function cry(){} 子類 protected function cry(){} 則會報錯
但是子類的存取權限可以放大,如:
父類private function cry(){} 子類 protected function cry(){} 可以正確執行
以上就是(進階篇)PHP物件導向三大特點學習(封裝、繼承、多態)的內容,更多相關內容請關注topic.alibabacloud.com(www.php.cn)!