經(jīng)驗(yàn)分享:讓PHP開(kāi)發(fā)者事半功倍的十大技巧
如果你使用一面大鏡子作為沖浪板會(huì)發(fā)生什么?或許你會(huì)在較短的時(shí)間內(nèi)征服海浪,但是你肯定從內(nèi)心深處明白,這不是沖浪的正確選擇。同樣的道理也適用于PHP編程,盡管這樣的類(lèi)比聽(tīng)起來(lái)有一些古怪。我們經(jīng)常聽(tīng)到有人試圖用一個(gè)周末多點(diǎn)的時(shí)間來(lái)學(xué)會(huì)PHP,但是請(qǐng)恕我直言,這是學(xué)習(xí)這門(mén)編程語(yǔ)言的一種非常糟糕的方式。為何說(shuō)學(xué)習(xí)PHP的過(guò)程有別于任何其它語(yǔ)言?
就其本質(zhì)而言,如果你掌握了以PHP語(yǔ)言“做事”的方式,那么在使用它時(shí)就會(huì)得心應(yīng)手,因此值得你去投入精力去了解這些方式。在PHP中,單純按照自己思路去解決問(wèn)題往往會(huì)是一種錯(cuò)誤的辦法。這并不是因?yàn)槟闶且粋€(gè)糟糕的程序員,而是因?yàn)槿绻阆雽?xiě)出好的可維護(hù)性強(qiáng)的代碼,有些標(biāo)準(zhǔn)技巧是你必須要使用的。下面讓我們一起看一下你需要知道的10大技巧。
1、如何正確的創(chuàng)建一個(gè)網(wǎng)站的Index頁(yè)面
創(chuàng)建每一個(gè)網(wǎng)站時(shí),建立網(wǎng)站的index頁(yè)面是首要做的事情之一。如果你是一個(gè)PHP新手,在編寫(xiě)index頁(yè)面時(shí)典型的做法是只對(duì)index頁(yè)面所需的內(nèi)容進(jìn)行編程,其它鏈接創(chuàng)建另一個(gè)頁(yè)面。不過(guò),如果想學(xué)習(xí)一種更高效的方式來(lái)實(shí)現(xiàn)PHP編程,可以采用“index.php?page=home”模式,許多網(wǎng)站都在采用這種模式。
2、使用Request Global Array抓取數(shù)據(jù)
實(shí)際上我們沒(méi)有任何理由使用$_GET和$_POST數(shù)組來(lái)抓取數(shù)值。$_REQUEST這個(gè)全局?jǐn)?shù)組能夠讓你獲取一個(gè)get或form請(qǐng)求。因此,多數(shù)情況下解析數(shù)據(jù)的更高效代碼大體如下:
$action = isset($_REQUEST['action']) ? $_REQUEST['action'] : 0;
3、利用var_dump進(jìn)行PHP代碼調(diào)試
如果你在尋找php調(diào)試技術(shù),我必須說(shuō)var_dump應(yīng)該是你要找的目標(biāo)。在顯示php信息方面這個(gè)命令可以滿(mǎn)足你的所有需要。而調(diào)試代碼的多數(shù)情況與得到PHP中的數(shù)值有關(guān)。
4、PHP處理代碼邏輯,Smarty處理展現(xiàn)層
Smarty是一個(gè)使用PHP寫(xiě)出來(lái)的模板PHP模板引擎,是目前業(yè)界最著名的PHP模板引擎之一。它分離了邏輯代碼和外在的內(nèi)容,提供了一種易于管理和使用的方法,用來(lái)將原本與HTML代碼混雜在一起PHP代碼邏輯分離。簡(jiǎn)單的講,目的就是要使PHP程序員同前端人員分離,使程序員改變程序的邏輯內(nèi)容不會(huì)影響到前端人員的頁(yè)面設(shè)計(jì),前端人員重新修改頁(yè)面不會(huì)影響到程序的程序邏輯,這在多人合作的項(xiàng)目中顯的尤為重要。
5、的確需要使用全局?jǐn)?shù)值時(shí),創(chuàng)建一個(gè)Config文件
動(dòng)輒創(chuàng)建全局?jǐn)?shù)值是一種糟糕的做法,不過(guò)有時(shí)候?qū)嶋H情況的確又需要這么做。對(duì)于數(shù)據(jù)庫(kù)表或數(shù)據(jù)庫(kù)連接信息使用全局?jǐn)?shù)值是一個(gè)不錯(cuò)的想法,但不要在你的PHP代碼中頻繁使用全局?jǐn)?shù)值。另外,更好的一種做法是把你的全局變量存放在一個(gè)config.php文件中。
6、如果未定義,禁止訪(fǎng)問(wèn)!
如果你正確的創(chuàng)建了頁(yè)面,那么任何其他人沒(méi)有理由訪(fǎng)問(wèn)index.php或home.php之外的index.php頁(yè)面。一旦index.php被訪(fǎng)問(wèn)后,你可以通過(guò)獲得變量的方式來(lái)打開(kāi)需要的頁(yè)面。你的index頁(yè)面應(yīng)該包含類(lèi)似的以下代碼:
- define('yourPage',1);
然后,其它頁(yè)面應(yīng)該包含:
- if (!defined('yourPage')) die('Access Denied');
這么做的目的是防止直接訪(fǎng)問(wèn)你的其它php頁(yè)面。這樣,任何試圖不通過(guò)index.php訪(fǎng)問(wèn)其它網(wǎng)頁(yè)的人,將得到“訪(fǎng)問(wèn)被拒絕”的消息。
7、創(chuàng)建一個(gè)數(shù)據(jù)庫(kù)類(lèi)
如果你正在進(jìn)行數(shù)據(jù)庫(kù)編程(在PHP中非常常見(jiàn)的任務(wù)),一個(gè)不錯(cuò)的想法是創(chuàng)建一個(gè)數(shù)據(jù)庫(kù)類(lèi)來(lái)處理任何數(shù)據(jù)庫(kù)管理功能。示例代碼如下:
- public function dbExec($query)
- {
- $result = $this->db->exec($query);
- if (PEAR::isError($result)) errorRedirect($result->getMessage(), true);
- else return $result;
- }
這個(gè)函數(shù)僅接收一個(gè)查詢(xún)語(yǔ)句并對(duì)其執(zhí)行。它還處理可能出現(xiàn)的任何錯(cuò)誤。你還可以在這兒包含審核代碼,不過(guò)我更喜歡使用一個(gè)類(lèi)似的審核函數(shù):
- // checks if arguments given are integer values not less than 0 - has multiple arguments function sanitizeInput() { $numargs = func_num_args();
- $arg_list = func_get_args();
- for ($i = 0; $i < $numargs; $i++)
- {
- if (!is_numeric($arg_list[$i]) || $arg_list[$i] < 0) errorRedirect("Unexpected variable value", true);
- }
- }
8、一個(gè)php文件處理輸入,一個(gè)class.php文件處理具體功能
讓代碼變得混亂的一個(gè)重要方法是:獲取用戶(hù)輸入后,將其重定向到其它函數(shù)來(lái)進(jìn)行處理。原理非常簡(jiǎn)單,php文件獲得我們需要的任何輸入,然后將其執(zhí)行重定向到類(lèi)文件中的一個(gè)函數(shù)。舉例來(lái)講,假設(shè)有一個(gè)類(lèi)似“index.php?page=profile&action=display”的 URL。由profile.php來(lái)檢索該網(wǎng)址并得到操作是“display”。然后使用一個(gè)簡(jiǎn)單的switch函數(shù),我們來(lái)執(zhí)行真正的顯示函數(shù):
- require_once PROJECTROOT.'libs/messages.class.php';
- $message = new Message();
- switch ($action) { case 'display': $message->display();
- break;
- ...
如上所示,我使用了一個(gè)消息類(lèi),然后開(kāi)始進(jìn)行switch檢查。$message只是被類(lèi)中的調(diào)用函數(shù)使用的一個(gè)對(duì)象。
9、了解你的SQL語(yǔ)句,并總是對(duì)其審查(Sanitize)
正如我以前所提到的,任何php網(wǎng)站中最重要的部分有99%的可能是數(shù)據(jù)庫(kù)。因此,你需要非常熟悉如何正確的使用sql。學(xué)會(huì)關(guān)聯(lián)表和更多高級(jí)技術(shù)。下面我將展示一個(gè)使用MySQL的函數(shù)示例,并使用本文第7條函數(shù)進(jìn)行審查。
- private function getSentMessages($id)
- { $this->util->sanitizeInput($id);
- $pm_table = $GLOBALS['config']['privateMsg'];
- $users = $GLOBALS['config']['users'];
- $sql = "SELECT PM.*, USR.username as name_sender FROM $pm_table PM,
- $users USR WHERE id_sender = '$id' AND sender_purge = FALSE AND USR.id = PM.id_receiver AND is_read = TRUE ORDER BY date_sent DESC";
- $result = $this->dbQueryAll($sql);
- return $result;
- }
首先,我們對(duì)用戶(hù)輸入進(jìn)行檢查(通過(guò)一個(gè)GET變量傳遞消息id),然后我們執(zhí)行我們的SQL命令。注意這兒SQL的用法。你需要了解如何使用別名和關(guān)聯(lián)表。
10、當(dāng)你只需要一個(gè)對(duì)象時(shí),使用單例模式
在PHP中相當(dāng)常見(jiàn)的一種情形時(shí),我們只需要?jiǎng)?chuàng)建一個(gè)對(duì)象一次,然后在我們的整個(gè)程序中使用它。一個(gè)很好的例子就是smarty變量,一旦被初始化后就可以在任何地方使用。這種情形的一個(gè)很好實(shí)現(xiàn)方案就是單例模式。示例代碼如下:
- function smartyObject()
- {
- if ($GLOBALS['config']['SmartyObj'] == 0)
- {
- $smarty = new SmartyGame();
- $GLOBALS['config']['SmartyObj'] = $smarty;
- }
- else $smarty = $GLOBALS['config']['SmartyObj'];
- return $smarty;
- }
注意,我們擁有一個(gè)全局smarty變量(該示例中它在config.php中被初始化),如果它的值為0,我們將創(chuàng)建一個(gè)新smarty對(duì)象。否則,它意味著該對(duì)象已經(jīng)被創(chuàng)建,我們只需要返回它。
【編輯推薦】