php 面向?qū)ο?/h1>
在面向?qū)ο蟮某绦蛟O(shè)計(jì)(英語(yǔ):object-oriented programming,縮寫:oop)中,對(duì)象是一個(gè)由信息及對(duì)信息進(jìn)行處理的描述所組成的整體,是對(duì)現(xiàn)實(shí)世界的抽象。
在現(xiàn)實(shí)世界里我們所面對(duì)的事情都是對(duì)象,如計(jì)算機(jī)、電視機(jī)、自行車等。
對(duì)象的主要三個(gè)特性:
- 對(duì)象的行為:可以對(duì) 對(duì)象施加那些操作,開(kāi)燈,關(guān)燈就是行為。
- 對(duì)象的形態(tài):當(dāng)施加那些方法是對(duì)象如何響應(yīng),顏色,尺寸,外型。
- 對(duì)象的表示:對(duì)象的表示就相當(dāng)于身份證,具體區(qū)分在相同的行為與狀態(tài)下有什么不同。
比如 animal(動(dòng)物) 是一個(gè)抽象類,我們可以具體到一只狗跟一只羊,而狗跟羊就是具體的對(duì)象,他們有顏色屬性,可以寫,可以跑等行為狀態(tài)。
1. 面向?qū)ο髢?nèi)容
- 類 ? 定義了一件事物的抽象特點(diǎn)。類的定義包含了數(shù)據(jù)的形式以及對(duì)數(shù)據(jù)的操作。
- 對(duì)象 ? 是類的范例。
- 成員變量 ? 定義在類內(nèi)部的變量。該變量的值對(duì)外是不可見(jiàn)的,但是可以通過(guò)成員函數(shù)訪問(wèn),在類被范例化為對(duì)象后,該變量即可稱為對(duì)象的屬性。
- 成員函數(shù) ? 定義在類的內(nèi)部,可用于訪問(wèn)對(duì)象的數(shù)據(jù)。
- 繼承 ? 繼承性是子類自動(dòng)共享父類數(shù)據(jù)結(jié)構(gòu)和方法的機(jī)制,這是類之間的一種關(guān)系。在定義和實(shí)現(xiàn)一個(gè)類的時(shí)候,可以在一個(gè)已經(jīng)存在的類的基礎(chǔ)之上來(lái)進(jìn)行,把這個(gè)已經(jīng)存在的類所定義的內(nèi)容作為自己的內(nèi)容,并加入若干新的內(nèi)容。
- 父類 ? 一個(gè)類被其他類繼承,可將該類稱為父類,或基類,或超類。
- 子類 ? 一個(gè)類繼承其他類稱為子類,也可稱為派生類。
- 多態(tài) ? 多態(tài)性是指相同的函數(shù)或方法可作用于多種類型的對(duì)象上并獲得不同的結(jié)果。不同的對(duì)象,收到同一消息可以產(chǎn)生不同的結(jié)果,這種現(xiàn)象稱為多態(tài)性。
- 重載 ? 簡(jiǎn)單說(shuō),就是函數(shù)或者方法有同樣的名稱,但是參數(shù)列表不相同的情形,這樣的同名不同參數(shù)的函數(shù)或者方法之間,互相稱之為重載函數(shù)或者方法。
- 抽象性 ? 抽象性是指將具有一致的數(shù)據(jù)結(jié)構(gòu)(屬性)和行為(操作)的對(duì)象抽象成類。一個(gè)類就是這樣一種抽象,它反映了與應(yīng)用有關(guān)的重要性質(zhì),而忽略其他一些無(wú)關(guān)內(nèi)容。任何類的劃分都是主觀的,但必須與具體的應(yīng)用有關(guān)。
- 封裝 ? 封裝是指將現(xiàn)實(shí)世界中存在的某個(gè)客體的屬性與行為綁定在一起,并放置在一個(gè)邏輯單元內(nèi)。
- 構(gòu)造函數(shù) ? 主要用來(lái)在創(chuàng)建對(duì)象時(shí)初始化對(duì)象, 即為對(duì)象成員變量賦初始值,總與new運(yùn)算符一起使用在創(chuàng)建對(duì)象的語(yǔ)句中。
- 析構(gòu)函數(shù) ? 析構(gòu)函數(shù)(destructor) 與構(gòu)造函數(shù)相反,當(dāng)對(duì)象結(jié)束其生命周期時(shí)(例如對(duì)象所在的函數(shù)已調(diào)用完畢),系統(tǒng)自動(dòng)執(zhí)行析構(gòu)函數(shù)。析構(gòu)函數(shù)往往用來(lái)做"清理善后" 的工作(例如在建立對(duì)象時(shí)用new開(kāi)辟了一片內(nèi)存空間,應(yīng)在退出前在析構(gòu)函數(shù)中用delete釋放)。
下圖中我們通過(guò) car 類 創(chuàng)建了三個(gè)對(duì)象:mercedes, bmw, 和 audi。
$mercedes = new car (); $bmw = new car (); $audi = new car ();
2. php 類定義
php 定義類通常語(yǔ)法格式如下:
class phpclass { var $var1; var $var2 = "constant string"; function myfunc ($arg1, $arg2) { [..] } [..] }
解析如下:
- 類使用 class 關(guān)鍵字后加上類名定義。
- 類名后的一對(duì)大括號(hào)({})內(nèi)可以定義變量和方法。
- 類的變量使用 var 來(lái)聲明, 變量也可以初始化值。
- 函數(shù)定義類似 php 函數(shù)的定義,但函數(shù)只能通過(guò)該類及其范例化的對(duì)象訪問(wèn)。
范例
class site { /* 成員變量 */ var $url; var $title; /* 成員函數(shù) */ function seturl($par){ $this--->url = $par; } function geturl(){ echo $this->url . php_eol; } function settitle($par){ $this->title = $par; } function gettitle(){ echo $this->title . php_eol; } } ?>
變量 $this 代表自身的對(duì)象。
php_eol 為換行符。
3. php 中創(chuàng)建對(duì)象
類創(chuàng)建后,我們可以使用 new 運(yùn)算符來(lái)范例化該類的對(duì)象:
$yapf = new site; $taobao = new site; $google = new site;
以上代碼我們創(chuàng)建了三個(gè)對(duì)象,三個(gè)對(duì)象各自都是獨(dú)立的,接下來(lái)我們來(lái)看看如何訪問(wèn)成員方法與成員變量。
調(diào)用成員方法
在范例化對(duì)象后,我們可以使用該對(duì)象調(diào)用成員方法,該對(duì)象的成員方法只能操作該對(duì)象的成員變量:
// 調(diào)用成員函數(shù),設(shè)置標(biāo)題和url $yapf->settitle( "碩編程教程" ); $taobao->settitle( "淘寶" ); $google->settitle( "google 搜索" ); $yapf->seturl( '' ); $taobao->seturl( 'www.taobao.com' ); $google->seturl( 'www.google.com' ); // 調(diào)用成員函數(shù),獲取標(biāo)題和url $yapf->gettitle(); $taobao->gettitle(); $google->gettitle(); $yapf->geturl(); $taobao->geturl(); $google->geturl();
完整代碼如下:
class site { /* 成員變量 */ var $url; var $title; /* 成員函數(shù) */ function seturl($par){ $this--->url = $par; } function geturl(){ echo $this->url . php_eol; } function settitle($par){ $this->title = $par; } function gettitle(){ echo $this->title . php_eol; } } $yapf = new site; $taobao = new site; $google = new site; // 調(diào)用成員函數(shù),設(shè)置標(biāo)題和url $yapf->settitle( "碩編程教程" ); $taobao->settitle( "淘寶" ); $google->settitle( "google 搜索" ); $yapf->seturl( '' ); $taobao->seturl( 'www.taobao.com' ); $google->seturl( 'www.google.com' ); // 調(diào)用成員函數(shù),獲取標(biāo)題和url $yapf->gettitle(); $taobao->gettitle(); $google->gettitle(); $yapf->geturl(); $taobao->geturl(); $google->geturl(); ?>
執(zhí)行以上代碼,輸出結(jié)果為:
碩編程教程
淘寶
google 搜索
www.taobao.com
www.google.com
4. php 構(gòu)造函數(shù)
構(gòu)造函數(shù)是一種特殊的方法。主要用來(lái)在創(chuàng)建對(duì)象時(shí)初始化對(duì)象, 即為對(duì)象成員變量賦初始值,在創(chuàng)建對(duì)象的語(yǔ)句中與 new 運(yùn)算符一起使用。
php 5 允許開(kāi)發(fā)者在一個(gè)類中定義一個(gè)方法作為構(gòu)造函數(shù),語(yǔ)法格式如下:
void __construct ([ mixed $args [, $... ]] )
在上面的例子中我們就可以通過(guò)構(gòu)造方法來(lái)初始化 $url 和 $title 變量:
function __construct( $par1, $par2 ) { $this->url = $par1; $this->title = $par2; }
現(xiàn)在我們就不需要再調(diào)用 settitle 和 seturl 方法了:
$yapf = new site('', '碩編程教程'); $taobao = new site('www.taobao.com', '淘寶'); $google = new site('www.google.com', 'google 搜索'); // 調(diào)用成員函數(shù),獲取標(biāo)題和url $yapf->gettitle(); $taobao->gettitle(); $google->gettitle(); $yapf->geturl(); $taobao->geturl(); $google->geturl();
5. 析構(gòu)函數(shù)
析構(gòu)函數(shù)(destructor) 與構(gòu)造函數(shù)相反,當(dāng)對(duì)象結(jié)束其生命周期時(shí)(例如對(duì)象所在的函數(shù)已調(diào)用完畢),系統(tǒng)自動(dòng)執(zhí)行析構(gòu)函數(shù)。
php 5 引入了析構(gòu)函數(shù)的概念,這類似于其它面向?qū)ο蟮恼Z(yǔ)言,其語(yǔ)法格式如下:
void __destruct ( void )
范例
class mydestructableclass { function __construct() { print "構(gòu)造函數(shù)\n"; $this--->name = "mydestructableclass"; } function __destruct() { print "銷毀 " . $this->name . "\n"; } } $obj = new mydestructableclass(); ?>
執(zhí)行以上代碼,輸出結(jié)果為:
構(gòu)造函數(shù) 銷毀 mydestructableclass
6. 繼承
php 使用關(guān)鍵字 extends 來(lái)繼承一個(gè)類,php 不支持多繼承,格式如下:
class child extends parent { // 代碼部分 }
范例
范例中 child_site 類繼承了 site 類,并擴(kuò)展了功能:
// 子類擴(kuò)展站點(diǎn)類別 class child_site extends site { var $category; function setcate($par){ $this--->category = $par; } function getcate(){ echo $this->category . php_eol; } }
7. 方法重寫
如果從父類繼承的方法不能滿足子類的需求,可以對(duì)其進(jìn)行改寫,這個(gè)過(guò)程叫方法的覆蓋(override),也稱為方法的重寫。
范例中重寫了 geturl 與 gettitle 方法:
function geturl() { echo $this->url . php_eol; return $this->url; } function gettitle(){ echo $this->title . php_eol; return $this->title; }
8. 訪問(wèn)控制
php 對(duì)屬性或方法的訪問(wèn)控制,是通過(guò)在前面添加關(guān)鍵字 public(公有),protected(受保護(hù))或 private(私有)來(lái)實(shí)現(xiàn)的。
- public(公有):公有的類成員可以在任何地方被訪問(wèn)。
- protected(受保護(hù)):受保護(hù)的類成員則可以被其自身以及其子類和父類訪問(wèn)。
- private(私有):私有的類成員則只能被其定義所在的類訪問(wèn)。
屬性的訪問(wèn)控制
類屬性必須定義為公有,受保護(hù),私有之一。如果用 var 定義,則被視為公有。
/** * define myclass */ class myclass { public $public = 'public'; protected $protected = 'protected'; private $private = 'private'; function printhello() { echo $this--->public; echo $this->protected; echo $this->private; } } $obj = new myclass(); echo $obj->public; // 這行能被正常執(zhí)行 echo $obj->protected; // 這行會(huì)產(chǎn)生一個(gè)致命錯(cuò)誤 echo $obj->private; // 這行也會(huì)產(chǎn)生一個(gè)致命錯(cuò)誤 $obj->printhello(); // 輸出 public、protected 和 private /** * define myclass2 */ class myclass2 extends myclass { // 可以對(duì) public 和 protected 進(jìn)行重定義,但 private 而不能 protected $protected = 'protected2'; function printhello() { echo $this->public; echo $this->protected; echo $this->private; } } $obj2 = new myclass2(); echo $obj2->public; // 這行能被正常執(zhí)行 echo $obj2->private; // 未定義 private echo $obj2->protected; // 這行會(huì)產(chǎn)生一個(gè)致命錯(cuò)誤 $obj2->printhello(); // 輸出 public、protected2 和 undefined ?>
方法的訪問(wèn)控制
類中的方法可以被定義為公有,私有或受保護(hù)。如果沒(méi)有設(shè)置這些關(guān)鍵字,則該方法默認(rèn)為公有。
/** * define myclass */ class myclass { // 聲明一個(gè)公有的構(gòu)造函數(shù) public function __construct() { } // 聲明一個(gè)公有的方法 public function mypublic() { } // 聲明一個(gè)受保護(hù)的方法 protected function myprotected() { } // 聲明一個(gè)私有的方法 private function myprivate() { } // 此方法為公有 function foo() { $this--->mypublic(); $this->myprotected(); $this->myprivate(); } } $myclass = new myclass; $myclass->mypublic(); // 這行能被正常執(zhí)行 $myclass->myprotected(); // 這行會(huì)產(chǎn)生一個(gè)致命錯(cuò)誤 $myclass->myprivate(); // 這行會(huì)產(chǎn)生一個(gè)致命錯(cuò)誤 $myclass->foo(); // 公有,受保護(hù),私有都可以執(zhí)行 /** * define myclass2 */ class myclass2 extends myclass { // 此方法為公有 function foo2() { $this->mypublic(); $this->myprotected(); $this->myprivate(); // 這行會(huì)產(chǎn)生一個(gè)致命錯(cuò)誤 } } $myclass2 = new myclass2; $myclass2->mypublic(); // 這行能被正常執(zhí)行 $myclass2->foo2(); // 公有的和受保護(hù)的都可執(zhí)行,但私有的不行 class bar { public function test() { $this->testprivate(); $this->testpublic(); } public function testpublic() { echo "bar::testpublic\n"; } private function testprivate() { echo "bar::testprivate\n"; } } class foo extends bar { public function testpublic() { echo "foo::testpublic\n"; } private function testprivate() { echo "foo::testprivate\n"; } } $myfoo = new foo(); $myfoo->test(); // bar::testprivate // foo::testpublic ?>
9. 接口
使用接口(interface),可以指定某個(gè)類必須實(shí)現(xiàn)哪些方法,但不需要定義這些方法的具體內(nèi)容。
接口是通過(guò) interface 關(guān)鍵字來(lái)定義的,就像定義一個(gè)標(biāo)準(zhǔn)的類一樣,但其中定義所有的方法都是空的。
接口中定義的所有方法都必須是公有,這是接口的特性。
要實(shí)現(xiàn)一個(gè)接口,使用 implements 操作符。類中必須實(shí)現(xiàn)接口中定義的所有方法,否則會(huì)報(bào)一個(gè)致命錯(cuò)誤。類可以實(shí)現(xiàn)多個(gè)接口,用逗號(hào)來(lái)分隔多個(gè)接口的名稱。
// 聲明一個(gè)'itemplate'接口 interface itemplate { public function setvariable($name, $var); public function gethtml($template); } // 實(shí)現(xiàn)接口 class template implements itemplate { private $vars = array(); public function setvariable($name, $var) { $this--->vars[$name] = $var; } public function gethtml($template) { foreach($this->vars as $name => $value) { $template = str_replace('{' . $name . '}', $value, $template); } return $template; } }
10. 常量
可以把在類中始終保持不變的值定義為常量。在定義和使用常量的時(shí)候不需要使用 $ 符號(hào)。
常量的值必須是一個(gè)定值,不能是變量,類屬性,數(shù)學(xué)運(yùn)算的結(jié)果或函數(shù)調(diào)用。
自 php 5.3.0 起,可以用一個(gè)變量來(lái)動(dòng)態(tài)調(diào)用類。但該變量的值不能為關(guān)鍵字(如 self,parent 或 static)。
范例
class myclass { const constant = '常量值'; function showconstant() { echo self::constant . php_eol; } } echo myclass::constant . php_eol; $classname = "myclass"; echo $classname::constant . php_eol; // 自 5.3.0 起 $class = new myclass(); $class--->showconstant(); echo $class::constant . php_eol; // 自 php 5.3.0 起 ?>
11. 抽象類
任何一個(gè)類,如果它里面至少有一個(gè)方法是被聲明為抽象的,那么這個(gè)類就必須被聲明為抽象的。
定義為抽象的類不能被范例化。
被定義為抽象的方法只是聲明了其調(diào)用方式(參數(shù)),不能定義其具體的功能實(shí)現(xiàn)。
繼承一個(gè)抽象類的時(shí)候,子類必須定義父類中的所有抽象方法;另外,這些方法的訪問(wèn)控制必須和父類中一樣(或者更為寬松)。例如某個(gè)抽象方法被聲明為受保護(hù)的,那么子類中實(shí)現(xiàn)的方法就應(yīng)該聲明為受保護(hù)的或者公有的,而不能定義為私有的。
abstract class abstractclass { // 強(qiáng)制要求子類定義這些方法 abstract protected function getvalue(); abstract protected function prefixvalue($prefix); // 普通方法(非抽象方法) public function printout() { print $this--->getvalue() . php_eol; } } class concreteclass1 extends abstractclass { protected function getvalue() { return "concreteclass1"; } public function prefixvalue($prefix) { return "{$prefix}concreteclass1"; } } class concreteclass2 extends abstractclass { public function getvalue() { return "concreteclass2"; } public function prefixvalue($prefix) { return "{$prefix}concreteclass2"; } } $class1 = new concreteclass1; $class1->printout(); echo $class1->prefixvalue('foo_') . php_eol; $class2 = new concreteclass2; $class2->printout(); echo $class2->prefixvalue('foo_') . php_eol; ?>
執(zhí)行以上代碼,輸出結(jié)果為:
concreteclass1 foo_concreteclass1 concreteclass2 foo_concreteclass2
此外,子類方法可以包含父類抽象方法中不存在的可選參數(shù)。
例如,子類定義了一個(gè)可選參數(shù),而父類抽象方法的聲明里沒(méi)有,則也是可以正常運(yùn)行的。
abstract class abstractclass { // 我們的抽象方法僅需要定義需要的參數(shù) abstract protected function prefixname($name); } class concreteclass extends abstractclass { // 我們的子類可以定義父類簽名中不存在的可選參數(shù) public function prefixname($name, $separator = ".") { if ($name == "pacman") { $prefix = "mr"; } elseif ($name == "pacwoman") { $prefix = "mrs"; } else { $prefix = ""; } return "{$prefix}{$separator} {$name}"; } } $class = new concreteclass; echo $class--->prefixname("pacman"), "\n"; echo $class->prefixname("pacwoman"), "\n"; ?>
輸出結(jié)果為:
mr. pacman mrs. pacwoman
12. static 關(guān)鍵字
聲明類屬性或方法為 static(靜態(tài)),就可以不范例化類而直接訪問(wèn)。
靜態(tài)屬性不能通過(guò)一個(gè)類已范例化的對(duì)象來(lái)訪問(wèn)(但靜態(tài)方法可以)。
由于靜態(tài)方法不需要通過(guò)對(duì)象即可調(diào)用,所以偽變量 $this 在靜態(tài)方法中不可用。
靜態(tài)屬性不可以由對(duì)象通過(guò) -> 操作符來(lái)訪問(wèn)。
自 php 5.3.0 起,可以用一個(gè)變量來(lái)動(dòng)態(tài)調(diào)用類。但該變量的值不能為關(guān)鍵字 self,parent 或 static。
class foo { public static $my_static = 'foo'; public function staticvalue() { return self::$my_static; } } print foo::$my_static . php_eol; $foo = new foo(); print $foo--->staticvalue() . php_eol; ?>
執(zhí)行以上程序,輸出結(jié)果為:
foo foo
13. final 關(guān)鍵字
php 5 新增了一個(gè) final 關(guān)鍵字。如果父類中的方法被聲明為 final,則子類無(wú)法覆蓋該方法。如果一個(gè)類被聲明為 final,則不能被繼承。
以下代碼執(zhí)行會(huì)報(bào)錯(cuò):
class baseclass { public function test() { echo "baseclass::test() called" . php_eol; } final public function moretesting() { echo "baseclass::moretesting() called" . php_eol; } } class childclass extends baseclass { public function moretesting() { echo "childclass::moretesting() called" . php_eol; } } // 報(bào)錯(cuò)信息 fatal error: cannot override final method baseclass::moretesting()
14. 調(diào)用父類構(gòu)造方法
php 不會(huì)在子類的構(gòu)造方法中自動(dòng)的調(diào)用父類的構(gòu)造方法。要執(zhí)行父類的構(gòu)造方法,需要在子類的構(gòu)造方法中調(diào)用 parent::__construct() 。
class baseclass { function __construct() { print "baseclass 類中構(gòu)造方法" . php_eol; } } class subclass extends baseclass { function __construct() { parent::__construct(); // 子類構(gòu)造方法不能自動(dòng)調(diào)用父類的構(gòu)造方法 print "subclass 類中構(gòu)造方法" . php_eol; } } class othersubclass extends baseclass { // 繼承 baseclass 的構(gòu)造方法 } // 調(diào)用 baseclass 構(gòu)造方法 $obj = new baseclass(); // 調(diào)用 baseclass、subclass 構(gòu)造方法 $obj = new subclass(); // 調(diào)用 baseclass 構(gòu)造方法 $obj = new othersubclass();
執(zhí)行以上程序,輸出結(jié)果為:
baseclass 類中構(gòu)造方法 baseclass 類中構(gòu)造方法 subclass 類中構(gòu)造方法 baseclass 類中構(gòu)造方法