在PHP中使用全局變量[2]
在PHP中使用全局變量【二】
第一部分
使用單件(Singletons)
解決函數參數問題的一種方法就是采用單件(Singletons)來代替函數參數。單件是一類特殊的對象,它們隻能實例化一次,而且含有一個靜態方法來返回對象的接口。下麵的例子演示了如何構建一個簡單的單件:
<?php
// Get instance of DBConnection
$db =& DBConnection::getInstance();
// Set user property on object
$db->user = 'sa';
// Set second variable (which points to the same instance)
$second =& DBConnection::getInstance();
// Should print 'sa'
echo $second->user;
Class DBConnection {
var $user;
function &getInstance() {
static $me;
if (is_object($me) == true) {
return $me;
}
$me = new DBConnection;
return $me;
}
function connect() {
// TODO
}
function query() {
// TODO
}
}
?>
上麵例子中最重要的部分是函數getInstance()。這個函數通過使用一個靜態變量$me來返回這個類的實例,從而確保了隻有一個DBConnection類的實例。
使用單件的好處就是我們不需要明確的傳遞一個對象,而是簡單的使用getInstance()方法來獲取到這個對象,就好像下麵這樣:
<?php
function test() {
$db = DBConnection::getInstance();
// Do something with the object
}
?>
然而使用單件也存在一係列的不足。首先,如果我們如何在一個類需要全局化多個對象呢?因為我們使用單件,所以這個不可能的(正如它的名字是單件一樣)。另外一個問題,單件不能使用個體測試來測試的,而且這也是完全不可能的,除非你引入所有的堆棧,而這顯然是你不想看到的。這也是為什麼單件不是我們理想中的解決方法的主要原因。
注冊模式
讓一些對象能夠被我們代碼中所有的組件使用到(譯者注:全局化對象或者數據)的最好的方法就是使用一個中央容器對象,用它來包含我們所有的對象。通常這種容器對象被人們稱為一個注冊器。它非常的靈活而且也非常的簡單。一個簡單的注冊器對象就如下所示:
<?php
Class Registry {
var $_objects = array();
function set($name, &$object) {
$this->_objects[$name] =& $object;
}
function &get($name) {
return $this->_objects[$name];
}
}
?>
使用注冊器對象的第一步就是使用方法set()來注冊一個對象:
<?php
$db = new DBConnection;
$settings = new Settings_XML;
$user = new User;
// Register objects
$registry =& new Registry;
$registry->set ('db', $db);
$registry->set ('settings', $settings);
$registry->set ('user', $user);
?>
現在我們的寄存器對象容納了我們所有的對象,我們指需要把這個注冊器對象傳遞給一個函數(而不是分別傳遞三個對象)。看下麵的例子:
<?php
function test(&$registry) {
$db =& $registry->get('db');
$settings =& $registry->get('settings');
$user =& $registry->get('user');
// Do something with the objects
}
?>
注冊器相比其他的方法來說,它的一個很大的改進就是當我們需要在我們的代碼中新增加一個對象的時候,我們不再需要改變所有的東西(譯者注:指程序中所有用到全局對象的代碼),我們隻需要在注冊器裏麵新注冊一個對象,然後它(譯者注:新注冊的對象)就立即可以在所有的組件中調用。
為了更加容易的使用注冊器,我們把它的調用改成單件模式(譯者注:不使用前麵提到的函數傳遞)。因為在我們的程序中隻需要使用一個注冊器,所以單件模式使非常適合這種任務的。在注冊器類裏麵增加一個新的方法,如下所示:
<?
function &getInstance() {
static $me;
if (is_object($me) == true) {
return $me;
}
$me = new Registry;
return $me;
}
?>
這樣它就可以作為一個單件來使用,比如:
<?php
$db = new DBConnection;
$settings = new Settings_XML;
$user = new User;
// Register objects
$registry =& Registry::getInstance();
$registry->set ('db', $db);
$registry->set ('settings', $settings);
$registry->set ('user', $user);
function test() {
$registry =& Registry::getInstance();
$db =& $registry->get('db');
$settings =& $registry->get('settings');
$user =& $registry->get('user');
// Do something with the objects
}
?>
正如你看到的,我們不需要把私有的東西都傳遞到一個函數,也不需要使用“global”關鍵字。所以注冊器模式是這個問題的理想解決方案,而且它非常的靈活。
請求封裝器
雖然我們的注冊器已經使“global”關鍵字完全多餘了,在我們的代碼中還是存在一種類型的全局變量:超級全局變量,比如變量$_POST,$_GET。雖然這些變量都非常標準,而且在你使用中也不會出什麼問題,但是在某些情況下,你可能同樣需要使用注冊器來封裝它們。
一個簡單的解決方法就是寫一個類來提供獲取這些變量的接口。這通常被稱為“請求封裝器”,下麵是一個簡單的例子:
<?php
Class Request {
var $_request = array();
function Request() {
// Get request variables
$this->_request = $_REQUEST;
}
function get($name) {
return $this->_request[$name];
}
}
?>
上麵的例子是一個簡單的演示,當然在請求封裝器(request wrapper)裏麵你還可以做很多其他的事情(比如:自動過濾數據,提供默認值等等)。
下麵的代碼演示了如何調用一個請求封裝器:
<?php
$request = new Request;
// Register object
$registry =& Registry::getInstance();
$registry->set ('request', &$request);
test();
function test() {
$registry =& Registry::getInstance();
$request =& $registry->get ('request');
// Print the 'name' querystring, normally it'd be $_GET['name']
echo htmlentities($request->get('name'));
}
?>
正如你看到的,現在我們不再依靠任何全局變量了,而且我們完全讓這些函數遠離了全局變量。
結論
在本文中,我們演示了如何從根本上移除代碼中的全局變量,而相應的用合適的函數和變量來替代。注冊模式是我最喜歡的設計模式之一,因為它是非常的靈活,而且它能夠防止你的代碼變得一塌煳塗。
另外,我推薦使用函數參數而不是單件模式來傳遞注冊器對象。雖然使用單件更加輕鬆,但是它可能會在以後出現一些問題,而且使用函數參數來傳遞也更加容易被人理解。
最後更新:2017-04-02 00:06:45