解析在PHP中使用全局變量的幾種方法
簡(jiǎn)介
即使開(kāi)發(fā)一個(gè)新的大型PHP程序,你也不可避免的要使用到全局?jǐn)?shù)據(jù),因?yàn)橛行?shù)據(jù)是需要用到你的代碼的不同部分的。一些常見(jiàn)的全局?jǐn)?shù)據(jù)有:程序設(shè)定類(lèi)、數(shù)據(jù)庫(kù)連接類(lèi)、用戶資料等等。有很多方法能夠使這些數(shù)據(jù)成為全局?jǐn)?shù)據(jù),其中最常用的就是使用“global”關(guān)鍵字申明,稍后在文章中我們會(huì)具體的講解到。
使用“global”關(guān)鍵字來(lái)申明全局?jǐn)?shù)據(jù)的唯一缺點(diǎn)就是它事實(shí)上是一種非常差的編程方式,而且經(jīng)常在其后導(dǎo)致程序中出現(xiàn)更大的問(wèn)題,因?yàn)槿謹(jǐn)?shù)據(jù)把你代碼中原本單獨(dú)的代碼段都聯(lián)系在一起了,這樣的后果就是如果你改變其中的某一部分代碼,可能就會(huì)導(dǎo)致其他部分出錯(cuò)。所以如果你的代碼中有很多全局的變量,那么你的整個(gè)程序必然是難以維護(hù)的。
本文將展示如何通過(guò)不同的技術(shù)或者設(shè)計(jì)模式來(lái)防止這種全局變量問(wèn)題。當(dāng)然,首先讓我們看看如何使用“global”關(guān)鍵字來(lái)進(jìn)行全局?jǐn)?shù)據(jù)以及它是如何工作的。
使用全局變量和“global”關(guān)鍵字
PHP默認(rèn)定義了一些“超級(jí)全局(Superglobals)”變量,這些變量自動(dòng)全局化,而且能夠在程序的任何地方中調(diào)用,比如$_GET和$_REQUEST等等。它們通常都來(lái)自數(shù)據(jù)或者其他外部數(shù)據(jù),使用這些變量通常是不會(huì)產(chǎn)生問(wèn)題的,因?yàn)樗麄兓旧鲜遣豢蓪?xiě)的。
但是你可以使用你自己的全局變量。使用關(guān)鍵字“global”你就可以把全局?jǐn)?shù)據(jù)導(dǎo)入到一個(gè)函數(shù)的局部范圍內(nèi)。如果你不明白“變量使用范圍”,請(qǐng)你自己參考PHP手冊(cè)上的相關(guān)說(shuō)明。
下面是一個(gè)使用“global”關(guān)鍵字的演示例子:
<?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”關(guān)鍵字是用來(lái)導(dǎo)入全局變量的。看起來(lái)它工作的很好,而且很簡(jiǎn)單,那么為什么我們還要擔(dān)心使用“global”關(guān)鍵字來(lái)定義全局?jǐn)?shù)據(jù)呢?
下面是三個(gè)很好的理由:
1、代碼重用幾乎是不可能的。
如果一個(gè)函數(shù)依賴于全局變量,那么想在不同的環(huán)境中使用這個(gè)函數(shù)幾乎是不可能的。另外一個(gè)問(wèn)題就是你不能提取出這個(gè)函數(shù),然后在其他的代碼中使用。
2、調(diào)試并解決問(wèn)題是非常困難的。
跟蹤一個(gè)全局變量比跟蹤一個(gè)非全局變量困難的多。一個(gè)全局變量可能會(huì)在一些不明顯的包含文件中被重新定義,即使你有一個(gè)非常好的程序編輯器(或者IDE)來(lái)幫助你,你也得花了幾個(gè)小時(shí)才能發(fā)現(xiàn)這個(gè)問(wèn)題所在。
3、理解這些代碼將是非常難的事情。
你很難弄清楚一個(gè)全局變量是從哪里來(lái)得,它是用來(lái)做什么的。在開(kāi)發(fā)的過(guò)程中,你可能會(huì)知道知道每一個(gè)全局變量,但大概一年之后,你可能會(huì)忘記其中至少一般的全局變量,這個(gè)時(shí)候你會(huì)為自己使用那么多全局變量而懊悔不已。
那么如果我們不使用全局變量,我們?cè)撌褂檬裁茨??下面讓我們看看一些解決方案。
使用函數(shù)參數(shù)
停止使用全局變量的一種方法就是簡(jiǎn)單的把變量作為函數(shù)的參數(shù)傳遞過(guò)去,如同下面所示:
<?php
$var = 'Hello World';
test ($var);
function test($var) {
echo $var;
}
?>
如果你僅僅只需要傳遞一個(gè)全局變量,那么這是一種非常優(yōu)秀甚至可以說(shuō)是杰出的解決方案,但是如果你要傳遞很多個(gè)值,那該怎么辦呢?
比如說(shuō),假如我們要使用一個(gè)數(shù)據(jù)庫(kù)類(lèi),一個(gè)程序設(shè)置類(lèi)和一個(gè)用戶類(lèi)。在我們代碼中,這三個(gè)類(lèi)在所有組件中都要用到,所以必須傳遞給每一個(gè)組件。如果我們使用函數(shù)參數(shù)的方法,我們不得不這樣:
<?php
$db = new DBConnection;
$settings = new Settings_XML;
$user = new User;
test($db, $settings, $user);
function test(&$db, &$settings, &$user) {
// Do something
}
?>
顯然,這是不值得的,而且一旦我們有新的對(duì)象需要加入,我們不得不為每一個(gè)函數(shù)增加多一個(gè)函數(shù)參數(shù)。因此我們需要用采用另外一種方式來(lái)解決。
使用單件(Singletons)解決函數(shù)參數(shù)問(wèn)題的一種方法就是采用單件(Singletons)來(lái)代替函數(shù)參數(shù)。單件是一類(lèi)特殊的對(duì)象,它們只能實(shí)例化一次,而且含有一個(gè)靜態(tài)方法來(lái)返回對(duì)象的接口。下面的例子演示了如何構(gòu)建一個(gè)簡(jiǎn)單的單件:
<?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
}
}
?>
上面例子中最重要的部分是函數(shù)getInstance()。這個(gè)函數(shù)通過(guò)使用一個(gè)靜態(tài)變量$me來(lái)返回這個(gè)類(lèi)的實(shí)例,從而確保了只有一個(gè)DBConnection類(lèi)的實(shí)例。
使用單件的好處就是我們不需要明確的傳遞一個(gè)對(duì)象,而是簡(jiǎn)單的使用getInstance()方法來(lái)獲取到這個(gè)對(duì)象,就好像下面這樣:
<?php
function test() {
$db = DBConnection::getInstance();
// Do something with the object
}
?>
然而使用單件也存在一系列的不足。首先,如果我們?nèi)绾卧谝粋€(gè)類(lèi)需要全局化多個(gè)對(duì)象呢?因?yàn)槲覀兪褂脝渭?,所以這個(gè)不可能的(正如它的名字是單件一樣)。另外一個(gè)問(wèn)題,單件不能使用個(gè)體測(cè)試來(lái)測(cè)試的,而且這也是完全不可能的,除非你引入所有的堆棧,而這顯然是你不想看到的。這也是為什么單件不是我們理想中的解決方法的主要原因。
注冊(cè)模式
讓一些對(duì)象能夠被我們代碼中所有的組件使用到(譯者注:全局化對(duì)象或者數(shù)據(jù))的最好的方法就是使用一個(gè)中央容器對(duì)象,用它來(lái)包含我們所有的對(duì)象。通常這種容器對(duì)象被人們稱為一個(gè)注冊(cè)器。它非常的靈活而且也非常的簡(jiǎn)單。一個(gè)簡(jiǎn)單的注冊(cè)器對(duì)象就如下所示:
<?php
Class Registry {
var $_objects = array();
function set($name, &$object) {
$this->_objects[$name] =& $object;
}
function &get($name) {
return $this->_objects[$name];
}
}
?>
使用注冊(cè)器對(duì)象的第一步就是使用方法set()來(lái)注冊(cè)一個(gè)對(duì)象:
<?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);
?>
現(xiàn)在我們的寄存器對(duì)象容納了我們所有的對(duì)象,我們指需要把這個(gè)注冊(cè)器對(duì)象傳遞給一個(gè)函數(shù)(而不是分別傳遞三個(gè)對(duì)象)??聪旅娴睦樱?BR>
<?php
function test(&$registry) {
$db =& $registry->get('db');
$settings =& $registry->get('settings');
$user =& $registry->get('user');
// Do something with the objects
}
?>
注冊(cè)器相比其他的方法來(lái)說(shuō),它的一個(gè)很大的改進(jìn)就是當(dāng)我們需要在我們的代碼中新增加一個(gè)對(duì)象的時(shí)候,我們不再需要改變所有的東西(譯者注:指程序中所有用到全局對(duì)象的代碼),我們只需要在注冊(cè)器里面新注冊(cè)一個(gè)對(duì)象,然后它(譯者注:新注冊(cè)的對(duì)象)就立即可以在所有的組件中調(diào)用。
為了更加容易的使用注冊(cè)器,我們把它的調(diào)用改成單件模式(譯者注:不使用前面提到的函數(shù)傳遞)。因?yàn)樵谖覀兊某绦蛑兄恍枰褂靡粋€(gè)注冊(cè)器,所以單件模式使非常適合這種任務(wù)的。在注冊(cè)器類(lèi)里面增加一個(gè)新的方法,如下所示:
<?
function &getInstance() {
static $me;
if (is_object($me) == true) {
return $me;
}
$me = new Registry;
return $me;
}
?>
這樣它就可以作為一個(gè)單件來(lái)使用,比如:
<?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
}
?>
正如你看到的,我們不需要把私有的東西都傳遞到一個(gè)函數(shù),也不需要使用“global”關(guān)鍵字。所以注冊(cè)器模式是這個(gè)問(wèn)題的理想解決方案,而且它非常的靈活。
請(qǐng)求封裝器
雖然我們的注冊(cè)器已經(jīng)使“global”關(guān)鍵字完全多余了,在我們的代碼中還是存在一種類(lèi)型的全局變量:超級(jí)全局變量,比如變量$_POST,$_GET。雖然這些變量都非常標(biāo)準(zhǔn),而且在你使用中也不會(huì)出什么問(wèn)題,但是在某些情況下,你可能同樣需要使用注冊(cè)器來(lái)封裝它們。
一個(gè)簡(jiǎn)單的解決方法就是寫(xiě)一個(gè)類(lèi)來(lái)提供獲取這些變量的接口。這通常被稱為“請(qǐng)求封裝器”,下面是一個(gè)簡(jiǎn)單的例子:
<?php
Class Request {
var $_request = array();
function Request() {
// Get request variables
$this->_request = $_REQUEST;
}
function get($name) {
return $this->_request[$name];
}
}
?>
上面的例子是一個(gè)簡(jiǎn)單的演示,當(dāng)然在請(qǐng)求封裝器(request wrapper)里面你還可以做很多其他的事情(比如:自動(dòng)過(guò)濾數(shù)據(jù),提供默認(rèn)值等等)。
下面的代碼演示了如何調(diào)用一個(gè)請(qǐng)求封裝器:
<?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'));
}
?>
正如你看到的,現(xiàn)在我們不再依靠任何全局變量了,而且我們完全讓這些函數(shù)遠(yuǎn)離了全局變量。
結(jié)論
在本文中,我們演示了如何從根本上移除代碼中的全局變量,而相應(yīng)的用合適的函數(shù)和變量來(lái)替代。注冊(cè)模式是我最喜歡的設(shè)計(jì)模式之一,因?yàn)樗欠浅5撵`活,而且它能夠防止你的代碼變得一塌糊涂。
另外,我推薦使用函數(shù)參數(shù)而不是單件模式來(lái)傳遞注冊(cè)器對(duì)象。雖然使用單件更加輕松,但是它可能會(huì)在以后出現(xiàn)一些問(wèn)題,而且使用函數(shù)參數(shù)來(lái)傳遞也更加容易被人理解。
- PHP正在進(jìn)行時(shí)-變量詳解及字符串動(dòng)態(tài)插入變量
- php解析http獲取的json字符串變量總是空白null
- php使用parse_str實(shí)現(xiàn)查詢字符串解析到變量中的方法
- php 變量定義方法
- php中static靜態(tài)變量的使用方法詳解
- php判斷變量類(lèi)型常用方法
- PHP中session變量的銷(xiāo)毀
- PHP 判斷常量,變量和函數(shù)是否存在
- PHP 變量類(lèi)型的強(qiáng)制轉(zhuǎn)換
- PHP中使用unset銷(xiāo)毀變量并內(nèi)存釋放問(wèn)題
- php 字符串中是否包含指定字符串的多種方法
- php變量與字符串的增刪改查操作示例
相關(guān)文章
PHP實(shí)現(xiàn)獲取ip地址的5種方法,以及插入用戶登錄日志操作示例
這篇文章主要介紹了PHP實(shí)現(xiàn)獲取ip地址的5種方法,以及插入用戶登錄日志操作,結(jié)合實(shí)例形式總結(jié)分析了php獲取訪客IP地址的5種常見(jiàn)操作方法,以及將用戶登陸信息寫(xiě)入登陸日志數(shù)據(jù)庫(kù)相關(guān)操作技巧,需要的朋友可以參考下2019-02-02PHP和JavaScrip分別獲取關(guān)聯(lián)數(shù)組的鍵值示例代碼
關(guān)聯(lián)數(shù)組的鍵值獲取,有很多方法,在本文為大家介紹下PHP和JavaScrip中時(shí)如何實(shí)現(xiàn)的,感興趣的朋友可以參考下2013-09-09php編程中echo用逗號(hào)和用點(diǎn)號(hào)連接的區(qū)別
本文通過(guò)幾個(gè)具體的例子.來(lái)認(rèn)清楚php編程中echo用逗號(hào)和用點(diǎn)號(hào)連接之前的區(qū)別.有需要的小伙伴可以參考下2016-03-03Php Image Resize圖片大小調(diào)整的函數(shù)代碼
php下可以將圖片大小調(diào)整的函數(shù)代碼,需要的朋友可以參考下。2011-01-01PHP 面向?qū)ο蟾倪M(jìn)后的一點(diǎn)說(shuō)明
學(xué)習(xí)php 面向?qū)ο蟮呐笥?,需要了解的東西。2008-10-1010個(gè)對(duì)初學(xué)者非常有用的PHP技巧
這篇文章主要為大家詳細(xì)介紹了10個(gè)對(duì)初學(xué)者非常有用的PHP技巧,這些PHP技巧適用于初學(xué)者,而不是那些已經(jīng)在使用MVC框架的人,感興趣的小伙伴們可以參考一下2016-04-04