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

開發(fā)大型PHP項(xiàng)目的方法

[摘要]開發(fā)大型PHP項(xiàng)目的方法這里介紹了在PHP中的面向?qū)ο缶幊?OOP,Object Oriented Programming)。將向你演示如何通過使用一些OOP的概念和PHP的技巧來減少編碼和提高質(zhì)量。祝你好運(yùn)! 面向?qū)ο缶幊痰母拍睿?不同的作者之間說法可能不一樣,但是一個(gè)OOP語言必須有以下幾方面...

開發(fā)大型PHP項(xiàng)目的方法這里介紹了在PHP中的面向?qū)ο缶幊?OOP,Object Oriented Programming)。將向你演示如何通過使用一些OOP的概念和PHP的技巧來減少編碼和提高質(zhì)量。祝你好運(yùn)!

面向?qū)ο缶幊痰母拍睿?
不同的作者之間說法可能不一樣,但是一個(gè)OOP語言必須有以下幾方面:

抽象數(shù)據(jù)類型和信息封裝
繼承
多態(tài)

在PHP中是通過類來完成封裝的:

代碼:

<?php
class Something {
// 在OOP類中,通常第一個(gè)字符為大寫
var $x;
function setX($v) {
// 方法開始為小寫單詞,然后使用大寫字母來分隔單詞,例如getValueOfArea()
$this->x=$v;
}
function getX() {
return $this->x;
}
}
?>



當(dāng)然你可以按自已的喜好進(jìn)行定義,但最好保持一種標(biāo)準(zhǔn),這樣會(huì)更有效。

數(shù)據(jù)成員在類中使用"var"聲明來定義,在給數(shù)據(jù)成員賦值之前,它們是沒有類型的。一個(gè)數(shù)據(jù)成員可以是一個(gè)整數(shù),一個(gè)數(shù)組,一個(gè)相關(guān)數(shù)組(associative array)或者是一個(gè)對(duì)象。

方法在類中被定義成函數(shù)形式,在方法中訪問類成員變量時(shí),你應(yīng)該使用$this->name,否則對(duì)一個(gè)方法來說,它只能是局部變量。

使用new操作符來創(chuàng)建一個(gè)對(duì)象:

$obj=new Something;

然后你可以使用成員函數(shù)通過:

$obj->setX(5);
$see=$obj->getX();

在這個(gè)例子中,setX成員函數(shù)將5賦值給對(duì)象的成員變量x(不是類的),然后getX返回它的值5。

你可以象:$obj->x=6那樣通過類引用方式來存取數(shù)據(jù)成員,這不是一個(gè)很好的OOP習(xí)慣。我強(qiáng)烈建議通過方法來存取成員變量。如果你把成員變量看成是不可處理的,并且只通過對(duì)象句柄來使用方法,你將是一 個(gè)好的OOP程序員。不幸的是,PHP不支持聲明私有成員變量,所以不良代碼在PHP中也是允許的。

繼承在PHP中很容易實(shí)現(xiàn),只要使用extend關(guān)鍵字。

代碼:

<?php
class Another extends Something {
var $y;
function setY($v) {
$this->y=$v;
}
function getY() {
return $this->y;
}
}
?>



"Another"類的對(duì)象現(xiàn)在擁有了父類(Something)的全部的數(shù)據(jù)成員及方法,而且還加上了自已的數(shù)據(jù)成 員和方法。

你可以使用

代碼:

$obj2=new Something;
$obj2->setX(6);
$obj2->setY(7);



PHP現(xiàn)在還不支持多重繼承,所以你不能從兩個(gè)或兩個(gè)以上類派生出新的類來。

你可以在派生類中重定義一個(gè)方法,如果我們?cè)?Another"類中重定義了getX方法,我們就不能使 用"Something"中的getX方法了。如果你在派生類中聲明了一個(gè)與基派同名的數(shù)據(jù)成員,那么當(dāng)你處理它時(shí), 它將“隱藏”基類的數(shù)據(jù)成員。

你可以在你的類中定義構(gòu)造函數(shù)。構(gòu)造函數(shù)是一個(gè)與類名同名的方法,當(dāng)你創(chuàng)建一個(gè)類的對(duì)象時(shí)會(huì)被調(diào) 用,例如:

代碼:

<?php
class Something {
var $x;
function Something($y) {
$this->x=$y;
}
function setX($v) {
$this->x=$v;
}
function getX() {
return $this->x;
}
}
?>



所以你可以創(chuàng)建一個(gè)對(duì)象,通過:

$obj=new Something(6);

構(gòu)造函數(shù)會(huì)自動(dòng)地把6賦值給數(shù)據(jù)變量x。構(gòu)造函數(shù)和方法都是普通的PHP函數(shù),所以你可以使用缺省參數(shù)。

function Something($x="3",$y="5")

接著:

$obj=new Something(); // x=3 and y=5
$obj=new Something(8); // x=8 and y=5
$obj=new Something(8,9); // x=8 and y=9

缺省參數(shù)使用C++的方式,所以你不能忽略Y的值,而給X一個(gè)缺省參數(shù),參數(shù)是從左到右賦值的,如果傳入的參數(shù)少于要求的參數(shù)時(shí),其作的將使用缺省參數(shù)。

當(dāng)一個(gè)派生類的對(duì)象被創(chuàng)建時(shí),只有它的構(gòu)造函數(shù)被調(diào)用,父類的構(gòu)造函數(shù)沒被調(diào)用,如果你想調(diào)用基類的構(gòu)造函數(shù),你必須要在派生類的構(gòu)造函數(shù)中顯示調(diào)用。可以這樣做是因?yàn)樵谂缮愔兴懈割惖姆椒ǘ际强捎玫摹?

代碼:

<?php
function Another() {
$this->y=5;
$this->Something();
//顯示調(diào)用基類構(gòu)造函數(shù)
}
?>



OOP的一個(gè)很好的機(jī)制是使用抽象類。抽象類是不能實(shí)例化,只能提供給派生類一個(gè)接口。設(shè)計(jì)者通常使用抽象類來強(qiáng)迫程序員從基類派生,這樣可以確保新的類包含一些期待的功能。在PHP中沒有標(biāo)準(zhǔn)的方法,但是:

如果你需要這個(gè)特性,可以通過定義基類,并在它的構(gòu)造函數(shù)后加上"die" 的調(diào)用,這樣就可以保證基類是不可實(shí)例化的,現(xiàn)在在每一個(gè)方法(接口)后面加上"die" 語句,所以,如果一個(gè)程序員在派生類中沒有覆蓋方法,將引發(fā)一個(gè)錯(cuò)誤。而且因?yàn)镻HP 是無類型的,你可能需要確認(rèn)一個(gè)對(duì)象是來自于你的基類的派生類,那么在基類中增加一個(gè)方法來實(shí)義類的身份(返回某種標(biāo)識(shí)id),并且在你接收到一個(gè)對(duì)象參數(shù)時(shí)校驗(yàn)這個(gè)值。當(dāng)然,如果一個(gè)邪惡不好的程序員在派生類中覆蓋了這個(gè)方法,這種方法就不起作用了,不過一般問題多發(fā)現(xiàn)在懶惰的程序員身上,而不是邪惡的程序員。

當(dāng)然,能夠讓基類對(duì)程序員無法看到是很好的,只要將接口打印出來做他們的工作就可以了。

在PHP中沒有析構(gòu)函數(shù)。

重載(與覆蓋不同)在PHP中不支持。在OOP中,你可以重載一個(gè)方法來實(shí)現(xiàn)兩個(gè)或重多的方法具有相同的名字,但是有不同數(shù)量或類型的參數(shù)(這要看語言)。PHP 是一種松散類型的語言,所以通過類型重載不起作用,然而通過參數(shù)的個(gè)數(shù)不同來重載也不起作用。

有時(shí)在OOP中重載構(gòu)造函數(shù)非常好,這樣你可以通過不同的方法創(chuàng)建對(duì)象(傳遞不同數(shù)量的參數(shù))。在PHP
中實(shí)現(xiàn)它的技巧是:

代碼:

<?php
class Myclass {
function Myclass() {
$name="Myclass".func_num_args();
$this->$name();
//注意$this->name()一般是錯(cuò)誤的,但是在這里$name是一個(gè)將被調(diào)用方法的名字
}
function Myclass1($x) {
code;
}
function Myclass2($x,$y) {
code;
}
}
?>



通過在類中的額外的處理,使用這個(gè)類對(duì)用戶是透明的:

$obj1=new Myclass('1'); //將調(diào)用Myclass1

$obj2=new Myclass('1','2'); //將調(diào)用Myclass2

有時(shí)這個(gè)非常好用。

多態(tài)
多態(tài)是對(duì)象的一種能力,它可以在運(yùn)行時(shí)刻根據(jù)傳遞的對(duì)象參數(shù),決定調(diào)用哪一個(gè)對(duì)象的方法。例如,如果你有一個(gè)figure的類,它定義了一個(gè)draw的方法。并且派生了circle和rectangle 類,在派生類中你覆蓋了draw方法,你可能還有一個(gè)函數(shù),它希望使用一個(gè)參數(shù)x,并且可以調(diào)用$x->draw() 。如果你有多態(tài)性,調(diào)用哪個(gè)draw方法就依賴于你傳遞給這個(gè)函數(shù)的對(duì)象類型。

多態(tài)性在象PHP這樣的解釋語言(想象一下一個(gè)C++編譯器生成這樣的代碼,你應(yīng)該調(diào)用哪一個(gè)方法?你也不知道你擁有的對(duì)象是什么類型的,好,這不是重點(diǎn))是非常容易和自然的。所以PHP當(dāng)然支持多態(tài)性。

代碼:

<?php
function niceDrawing($x) {
//假設(shè)這是Board類的一個(gè)方法
$x->draw();
}
$obj=new Circle(3,187);
$obj2=new Rectangle(4,5);
$board->niceDrawing($obj);
//將調(diào)用Circle的draw方法
$board->niceDrawing($obj2);
//將調(diào)用Rectangle的draw方法
?>



用PHP進(jìn)行面向?qū)ο缶幊?
一些"純化論者(purists)"可能會(huì)說PHP不是一個(gè)真正的面向?qū)ο蟮恼Z言,這是事實(shí)。PHP 是一個(gè)混合型語言,你可以使用OOP,也可以使用傳統(tǒng)的過程化編程。然而,對(duì)于大型項(xiàng)目,你可能想/需要在PHP 中使用純的OOP去聲明類,而且在你的項(xiàng)目只用對(duì)象和類。

隨著項(xiàng)目越來越大,使用OOP可能會(huì)有幫助,OOP代碼很容易維護(hù),容易理解和重用。這些就是軟件工程
的基礎(chǔ)。在基于web的項(xiàng)目中應(yīng)用這些概念就成為將來網(wǎng)站成功的關(guān)鍵。

PHP的高級(jí)OOP技術(shù)
在看過基本的OOP概念后,我就可以向你展示更高級(jí)的技術(shù):

序列化(Serializing)
PHP不支持永久對(duì)象,在OOP中永久對(duì)象是可以在多個(gè)應(yīng)用的引用中保持狀態(tài)和功能的對(duì)象,這意味著擁有將對(duì)象保存到一個(gè)文件或數(shù)據(jù)庫(kù)中的能力,而且可以在以后裝入對(duì)象。這就是所謂的序列化機(jī)制。PHP 擁有序列化方法,它可以通過對(duì)象進(jìn)行調(diào)用,序列化方法可以返回對(duì)象的字符串表示。然而,序列化只保存了對(duì)象的成員數(shù)據(jù)而不包話方法。

在PHP4中,如果你將對(duì)象序列化到字符串$s中,然后釋放對(duì)象,接著反序列化對(duì)象到$obj,你可以繼續(xù)使用對(duì)象的方法!我不建議這樣去做,因?yàn)?a)文檔中沒有保證這種行為在以后的版本中仍然可以使用。(b)這個(gè)可能導(dǎo)致一種誤解,在你把一個(gè)序列化后的版本保存到磁盤并退出腳本時(shí)。當(dāng)以后運(yùn)行這個(gè)腳本時(shí),你不能期待著在反序列化一個(gè)對(duì)象時(shí),對(duì)象的方法也會(huì)在那里,因?yàn)樽址硎靖揪筒话ǚ椒ā?

總而言之,PHP 進(jìn)行序列化對(duì)于保存對(duì)象的成員變量非常有用。(你也可以將相關(guān)數(shù)組和數(shù)組序列化到一個(gè)文件中)。

例子 :

代碼:

<?php
$obj=new Classfoo();
$str=serialize($obj);
//幾個(gè)月以后
//從磁盤中裝入str
$obj2=unserialize($str)
?>



你恢復(fù)了成員數(shù)據(jù),但是不包括方法(根據(jù)文檔所說)。這導(dǎo)致了只能通過類似于使用$obj2->x來存取成員變量(你沒有別的方法。┑奈ㄒ晦k法,所以不要在家里試它。

有一些辦法可以解決這個(gè)問題,我把它留著,因?yàn)閷?duì)這篇簡(jiǎn)潔的文章來說,他們太不好。

使用類進(jìn)行數(shù)據(jù)存儲(chǔ)
對(duì)于PHP和OOP一件非常好的事情就是,你可以很容易地定義一個(gè)類來操作某件事情,并且無論何時(shí)你想用的時(shí)候都可以調(diào)用相應(yīng)的類。假設(shè)你有一個(gè)HTML表單,用戶可以通過選擇產(chǎn)品ID號(hào)來選擇一個(gè)產(chǎn)品。在數(shù)據(jù)庫(kù)中有產(chǎn)品的信息,你想把產(chǎn)品顯示出來,顯示它的價(jià)格等等。你擁有不同類型的產(chǎn)品,并且同一個(gè)動(dòng)作可能對(duì)不同的產(chǎn)品具有不同的意思。例如,顯示一個(gè)聲音可能意味著播放它,但是對(duì)于其它種類的產(chǎn)品可能意味著顯示一個(gè)存在數(shù)據(jù)庫(kù)中的圖片。你可以使用OOP或PHP來減少編碼并提高質(zhì)量:

定義一個(gè)產(chǎn)品的類,定義它應(yīng)該有的方法(例如:顯示),然后定義對(duì)每一種類型的產(chǎn)品的類,從產(chǎn)品類派后出來(SoundItem類,ViewableItem類,等等),覆蓋在產(chǎn)品類中的方法,使它們按你的想法動(dòng)作。

根據(jù)數(shù)據(jù)庫(kù)中每一種產(chǎn)品的類型(type)字段給類命名,一個(gè)典型的產(chǎn)品表可能有(id, type, price, description, 等等字段)...然后在處理腳本中,你可以從數(shù)據(jù)庫(kù)中取出type值,然后實(shí)例化一個(gè)名為type的對(duì)象:


代碼:

<?php
$obj=new $type();
$obj->action();
?>



這是PHP的一個(gè)非常好的特性,你可以不用考慮對(duì)象的類型,調(diào)用$obj的顯示方法或其它的方法。使用這個(gè)技術(shù),你不需要修改腳本去增加一個(gè)新類型的對(duì)象,只是增加一個(gè)處理它的類。

這個(gè)功能很強(qiáng)大,只要定義方法,而不去考慮所有對(duì)象的類型,在不同的類中按不同的方法實(shí)現(xiàn)它們,然后在主腳本中對(duì)任意對(duì)象使用它們,沒有if...else,也不需要兩個(gè)程序員,只有高興。

現(xiàn)在你同意編程是容易的,維護(hù)是便宜的,可重用是真的嗎?

如果你管理一組程序員,分配工作就是很簡(jiǎn)單的了,每個(gè)人可能負(fù)責(zé)一個(gè)類型的對(duì)象和處理它的類。

可以通過這個(gè)技術(shù)實(shí)現(xiàn)國(guó)際化,根據(jù)用戶所選的語言字段應(yīng)用相應(yīng)的類就可以了,等等。


拷貝和克隆
當(dāng)你創(chuàng)建一個(gè)$obj的對(duì)象時(shí),你可以通過$obj2=$obj來拷貝對(duì)象,新的對(duì)象是$obj的一個(gè)拷貝(不是一個(gè)引用),所以它具有$obj在當(dāng)時(shí)的狀態(tài)。有時(shí)候,你不想這樣,你只是想生成一個(gè)象obj類一樣的一個(gè)新的對(duì)象,可以通過使用new語句來調(diào)用類的構(gòu)造函數(shù)。在PHP中也可以通過序列化,和一個(gè)基類來實(shí)現(xiàn),但所有的其它類都要從基類派生出來。


進(jìn)入危險(xiǎn)區(qū)域
當(dāng)你序列化一個(gè)對(duì)象,你會(huì)得到某種格式的字符串,如果你感興趣,你可以調(diào)究它,其中,字符串中有類的名字(太好了。,你可以把它取出來,象:

代碼:

<?php
$herring=serialize($obj);
$vec=explode(':',$herring);
$nam=str_replace("\"",'',$vec[2]);
?>



所以假設(shè)你創(chuàng)建了一個(gè)"Universe"的類,并且強(qiáng)制所有的類都必須從universe擴(kuò)展,你可以在universe中定義一個(gè)clone的方法,如下:

代碼:

<?php
class Universe {
function clone() {
$herring=serialize($this);
$vec=explode(':',$herring);
$nam=str_replace("\"",'',$vec[2]);
$ret=new $nam;
return $ret;
}
}
//然后
$obj=new Something();
//從Universe擴(kuò)展
$other=$obj->clone();
?>



你所得到的是一個(gè)新的Something類的對(duì)象,它同使用new方法,調(diào)用構(gòu)造函數(shù)創(chuàng)建出的對(duì)象一樣。我不知道這個(gè)對(duì)你是否有用,但是Universe類可以知道派生類的名字是一個(gè)好的經(jīng)驗(yàn)。想象是唯一的限制。 (出處:風(fēng)閃)