明輝手游網(wǎng)中心:是一個(gè)免費(fèi)提供流行視頻軟件教程、在線學(xué)習(xí)分享的學(xué)習(xí)平臺(tái)!

第8節(jié)--訪問方式 -- Classes and Objects in PHP5 [8]

[摘要]第八節(jié)--訪問方式PHP5的訪問方式允許限制對(duì)類成員的訪問. 這是在PHP5中新增的功能,但在許多面向?qū)ο笳Z言中都早已存在. 有了訪問方式,才能開發(fā)一個(gè)可靠的面向?qū)ο髴?yīng)用程序,并且構(gòu)建可重用的面向?qū)ο箢悗?像C++和Java一樣,PHP有三種訪問方式:public,private和protecte...
第八節(jié)--訪問方式

PHP5的訪問方式允許限制對(duì)類成員的訪問. 這是在PHP5中新增的功能,但在許多面向?qū)ο笳Z言中都早已存在. 有了訪問方式,才能開發(fā)一個(gè)可靠的面向?qū)ο髴?yīng)用程序,并且構(gòu)建可重用的面向?qū)ο箢悗?

像C++和Java一樣,PHP有三種訪問方式:public,private和protected. 對(duì)于一個(gè)類成員的訪問方式,可以是其中之一. 如果你沒有指明訪問方式,默認(rèn)地訪問方式為public. 你也可以為靜態(tài)成員指明一種訪問方式,將訪問方式放在static關(guān)鍵字之前(如public static).

Public成員可以被毫無限制地訪問.類外部的任何代碼都可以讀寫public屬性. 你可以從腳本的任何地方調(diào)用一個(gè)public方法. 在PHP的前幾個(gè)版本中,所有方法和屬性都是public, 這讓人覺得對(duì)象就像是結(jié)構(gòu)精巧的數(shù)組.

Private(私有)成員只在類的內(nèi)部可見. 你不能在一個(gè)private屬性所在的類方法之外改變或讀取它的值. 同樣地,只有在同一個(gè)類中的方法可以調(diào)用一個(gè)private方法. 繼承的子類也不能訪問父類中的private 成員.

要注意,類中的任何成員和類的實(shí)例都可以訪問private成員. 看例子6.8,equals方法將兩個(gè)widget進(jìn)行比較.==運(yùn)算符比較同一個(gè)類的兩個(gè)對(duì)象,但這個(gè)例子中每個(gè)對(duì)象實(shí)例都有唯一的ID.equals方法只比較name和price. 注意equals方法如何訪問另一個(gè)Widget實(shí)例的private屬性. Java和C都允許這樣的操作.

Listing 6.8 Private members
<?php    class Widget    {        private $name;        private $price;        private $id;        public function __construct($name, $price)        {            $this->name = $name;            $this->price = floatval($price);            $this->id = uniqid();        }        //checks if two widgets are the same 檢查兩個(gè)widget是否相同        public function equals($widget)        {            return(($this->name == $widget->name)AND                ($this->price == $widget->price));        }    }    $w1 = new Widget('Cog', 5.00);    $w2 = new Widget('Cog', 5.00);    $w3 = new Widget('Gear', 7.00);    //TRUE    if($w1->equals($w2))    {        print("w1 and w2 are the same<br>\n");    }    //FALSE    if($w1->equals($w3))    {        print("w1 and w3 are the same<br>\n");    }    //FALSE, == includes id in comparison    if($w1 == $w2) //不等,因?yàn)镮D不同    {        print("w1 and w2 are the same<br>\n");    } ?> 


如果你對(duì)面向?qū)ο缶幊滩皇煜?你可能想知道用private成員的目的是什么. 你可以回憶一下封裝和耦合的想法,這在本章開頭我們有討論過. Private成員有助于封裝數(shù)據(jù). 他們可以隱藏在一個(gè)類內(nèi)部而不被類外部的代碼接觸到. 同時(shí)他們還有助于實(shí)現(xiàn)松散的耦合. 如果數(shù)據(jù)結(jié)構(gòu)外的代碼不能直接訪問內(nèi)部屬性,那么就不會(huì)產(chǎn)生一個(gè)隱性的關(guān)聯(lián)性.

當(dāng)然,大部分private屬性仍然可以被外部代碼共享. 解決方法是用一對(duì)public方法,一個(gè)是get(獲取屬性的值),另一個(gè)是set(設(shè)置屬性的值). 構(gòu)造函數(shù)也接受屬性的初始值. 這使得成員間的交流通過一個(gè)狹窄的,經(jīng)過良好限定的接口來進(jìn)行. 這也提供改變傳遞給方法的值的機(jī)會(huì). 注意在例子6.8中,構(gòu)造函數(shù)如何強(qiáng)制使price成為一個(gè)float數(shù)(floadval()).

Protected(受保護(hù)的) 成員能被同個(gè)類中的所有方法和繼承出的類的中所有方法訪問到. Public屬性有違封裝的精神,因?yàn)樗鼈冊(cè)试S子類依賴于一個(gè)特定的屬性來書寫.protected方法則不會(huì)帶來這方面的擔(dān)憂.一個(gè)使用protected方法的子類需要很清楚它的父類的結(jié)構(gòu)才行.

例子6.9由例子6.8改進(jìn)而得到,包含了一個(gè)Widget的子類Thing. 注意Widget現(xiàn)在有一個(gè)叫作getName的protected方法. 如果Widget的實(shí)例試圖調(diào)用protected方法將會(huì)出錯(cuò): $w1->getName()產(chǎn)生了一個(gè)錯(cuò)誤. 但子類Thing中的getName方法可以調(diào)用這個(gè)protected方法.當(dāng)然對(duì)于證明Widget::getName方法是protected,這個(gè)例子顯得過于簡單. 在實(shí)際情況下,使用protected方法要依賴于對(duì)對(duì)象的內(nèi)部結(jié)構(gòu)的理解.

Listing 6.9 Protected members
<?php    class Widget    {        private $name;        private $price;        private $id;        public function __construct($name, $price)        {            $this->name = $name;            $this->price = floatval($price);            $this->id = uniqid();        }        //checks if two widgets are the same        public function equals($widget)        {            return(($this->name == $widget->name)AND                ($this->price == $widget->price));        }        protected function getName()        {            return($this->name);        }    }    class Thing extends Widget    {        private $color;        public function setColor($color)        {            $this->color = $color;        }        public function getColor()        {            return($this->color);        }        public function getName()        {            return(parent::getName());        }    }    $w1 = new Widget('Cog', 5.00);    $w2 = new Thing('Cog', 5.00);    $w2->setColor('Yellow');    //TRUE (still!) 結(jié)果仍然為真    if($w1->equals($w2))    {        print("w1 and w2 are the same<br>\n");    }    //print Cog 輸出 Cog    print($w2->getName()); ?> 




一個(gè)子類可能改變通過覆寫父類方法來改變方法的訪問方式,盡管如此,仍然有一些限制. 如果你覆寫了一個(gè)public類成員,他子類中必須保持public. 如果你覆寫了一個(gè)protected成員,它可保持protected或變成public.Private成員仍然只在當(dāng)前類中可見. 聲明一個(gè)與父類的private成員同名的成員將簡單地在當(dāng)前類中建立一個(gè)與原來不同的成員. 因此,在技術(shù)上你不能覆寫一個(gè)private成員.

Final關(guān)鍵字是限制訪問成員方法的另一個(gè)方法. 子類不能覆寫父類中標(biāo)識(shí)為final的方法. Final關(guān)鍵字不能用于屬性.

//haohappy注:PHP5的面向?qū)ο竽P腿匀徊粔蛲晟?如final不像Java中那樣對(duì)Data,Method甚至Class都可以用.