即使開發一個新的大型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
}
?>
顯然,這是不值得的,而且一旦我們有新的對象需要加入,我們不得不為每一個函數增加多一個函數參數。因此我們需要用采用另外一種方式來解決。