1. 程式人生 > >php類的封裝、繼承和多型的簡單理解

php類的封裝、繼承和多型的簡單理解

面象對向的三大特點:封裝性、繼承性、多型性 首先簡單理解一下抽象:
我們在前面定義一個類的時候,實際上就是把一類事物共有的屬性和行為提取出來,形成一個物理模型(模版),這種研究問題的方法稱為抽象


一、封裝性 
封裝就是把抽取出來的資料和對資料的操作封裝在一起,資料被保護在內部,程式的其他部分只有被授權的操作(方法)才能對資料進行操作。 
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(){} 可以正確執行 
擴充套件: 
方法過載(overload) 
基本概念:函式名相同,但引數的個數或引數的型別不同,達到呼叫同一個函式,可以區分不同的函式 
在PHP5中雖然也支援過載,但是和其它語言還是有很大區別的,php中不能定義多個同名函式 
PHP5中提供了強大的“魔術”函式,使用這些魔術函式,我們可以做到函式過載, 
這裡我們要到到 __call,當一個物件調一個方法時,而該方法不存在,則程式會自動呼叫__call 
【官方不推薦使用】 
PHP中有以下幾個魔術常量:__LINE__ __FILE__ __DIR__ __FUNCTION__ __CLASS__ 等 
例: 
複製程式碼 程式碼如下:


<?php 
class A{ 
function test1($p){ 
echo 'test1<br/>'; 

function test2($p){ 
echo 'test2<br/>'; 

function __call($method,$p){ 
//這裡$p為陣列,上面兩個變數名可自定義 
if($method == 'test'){ 
if(count($p)==1){ 
$this->test1($p); 
} else if(count($p)==2){ 
$this->test2($p); 




$a=new A(); 
$a->test(5); 
$a->test(3,5); 
?>