即使開發一個新的大型PHP程序,你也不可避免的要使用到全域數據,因為有些數據是需要用到你的程式碼的不同部分的。一些常見的全域資料有:程式設定類別、資料庫連線類別、使用者資料等等。有很多方法能夠使這些數據成為全域數據,其中最常用的就是使用「global」關鍵字申明,稍後在文章中我們會具體的講解到。
使用「global」關鍵字來申明全域資料的唯一缺點就是它事實上是一種非常差的程式設計方式,而且經常在其後導致程式中出現更大的問題,因為全域資料把你程式碼中原本單獨的程式碼段都連結在一起了,這樣的後果就是如果你改變其中的某一部分程式碼,可能會導致其他部分出錯。所以如果你的程式碼中有很多全域的變量,那麼你的整個程式必然是難以維護的。
本文將展示如何透過不同的技術或設計模式來防止這種全域變數問題。當然,首先讓我們看看如何使用「global」關鍵字來進行全域資料以及它是如何運作的。使用全域變數和「global」關鍵字
PHP預設定義了一些「超級全域(Superglobals)」變量,這些變數會自動全域化,而且能夠在程式的任何地方中調用,例如$_GET和$_REQUEST等等。它們通常都來自數據或其他外部數據,使用這些變數通常是不會產生問題的,因為他們基本上是不可寫的。
但是你可以使用你自己的全域變數。使用關鍵字「global」你就可以把全域資料匯入到一個函數的局部範圍內。如果你不明白“變數使用範圍”,請你自己參考PHP手冊上的相關說明。
以下是使用「global」關鍵字的示範範例:
以下為引用的內容:
<?php
$my_var = 'Hello World';
test_global();
function test_global() {
// Now in local scope
// the $my_var variable doesn't exist
// Produces error: "Undefined variable: my_var"
echo $my_var;
// Now let's important the variable
global $my_var;
// Works:
echo $my_var;
}
?>
如你在上面的例子中看到的一樣,「global」關鍵字是用來匯入全域變數的。看起來它運作的很好,而且很簡單,那麼為什麼我們還要擔心使用「global」關鍵字來定義全域資料呢?
以下是三個很好的理由:
1、 程式碼重用幾乎是不可能的
如果一個函數依賴全域變量,那麼想在不同的環境中使用這個函數幾乎是不可能的。另外一個問題就是你不能提取出這個函數,然後在其他的程式碼中使用。
2、 除錯並解決問題是非常困難的
追蹤一個全域變數比追蹤一個非全域變數困難的多。一個全域變數可能會在一些不明顯的包含檔案中被重新定義,即使你有一個非常好的程式編輯器(或IDE)來幫助你,你也得花了幾個小時才能發現這個問題所在。
3. 理解這些程式碼將是非常難的事情
你很難弄清楚一個全域變數是從哪裡來,它是用來做什麼的。在開發的過程中,你可能會知道知道每一個全域變量,但大概一年之後,你可能會忘記其中至少一般的全域變量,這個時候你會為自己使用那麼多全域變量而懊悔不已。
那如果我們不使用全域變量,我們該使用什麼呢?下面讓我們來看看一些解決方案。
使用函數參數
停止使用全域變數的一種方法就是簡單的把變數當作函數的參數傳遞過去,如同下面所示:
以下為引用的內容:
<?php
$var = 'Hello World';
test ($var);
function test($var) {
echo $var;
}
?>
如果你只需要傳遞一個全域變量,那麼這是一個非常優秀甚至可以說是傑出的解決方案,但是如果你要傳遞很多個值,那該怎麼辦呢?
比如說,如果我們要使用一個資料庫類,一個程式設定類別和一個使用者類別。在我們程式碼中,這三個類別在所有元件中都要用到,所以必須傳遞給每一個元件。如果我們使用函數參數的方法,我們不得不這樣:
以下為引用的內容:
<?php
$db = new DBConnection;
$settings = new Settings_XML;
$user = new User;
test($db, $settings, $user);
function test(&$db, &$settings, &$user) {
// Do something
}
?>
顯然,這是不值得的,而且一旦我們有新的物件需要加入,我們就必須為每一個函數增加多一個函數參數。因此我們需要用採用另一種方式來解決。