php中global 的运用,php 中全局變量global 的使用

簡介即使開發一個新的大型PHP程序,你也不可避免的要使用到全局數據,因為有些數據是需要用到你的代碼的不同部分的。一些常見的全局數據有:程序設定類、數 據庫連接類、用戶資料等等。有很多方法能夠使這些數據成為全局數據,其中最常用的就是使用“global”關鍵字申明,稍后在文章中我們會具體的講解到。

使用“global”關鍵字來申明全局數據的唯一缺點就是它事實上是一種非常差的編程方式,而且經常在其后導致程序中出現更大的問題,因為全局數據把你 代碼中原本單獨的代碼段都聯系在一起了,這樣的后果就是如果你改變其中的某一部分代碼,可能就會導致其他部分出錯。所以如果你的代碼中有很多全局的變量, 那么你的整個程序必然是難以維護的。

本文將展示如何通過不同的技術或者設計模式來防止這種全局變量問題。當然,首先讓我們看看如何使用“global”關鍵字來進行全局數據以及它是如何工作的。

使用全局變量和“global”關鍵字

PHP默認定義了一些“超級全局(Superglobals)”變量,這些變量自動全局化,而且能夠在程序的任何地方中調用,比如$_GET和$ _REQUEST等等。它們通常都來自數據或者其他外部數據,使用這些變量通常是不會產生問題的,因為他們基本上是不可寫的。

但是你可以使用你自己的全局變量。使用關鍵字“global”你就可以把全局數據導入到一個函數的局部范圍內。如果你不明白“變量使用范圍”,請你自己參考PHP手冊上的相關說明。

下面是一個使用“global”關鍵字的演示例子:

$my_var='Hello World';test_global();

functiontest_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 variableglobal$my_var;// Works:echo$my_var;

}?>

正如你在上面的例子中看到的一樣,“global”關鍵字是用來導入全局變量的。看起來它工作的很好,而且很簡單,那么為什么我們還要擔心使用“global”關鍵字來定義全局數據呢?

下面是三個很好的理由:

1、   代碼重用幾乎是不可能的。

如果一個函數依賴於全局變量,那么想在不同的環境中使用這個函數幾乎是不可能的。另外一個問題就是你不能提取出這個函數,然后在其他的代碼中使用。

2、   調試並解決問題是非常困難的。

跟蹤一個全局變量比跟蹤一個非全局變量困難的多。一個全局變量可能會在一些不明顯的包含文件中被重新定義,即使你有一個非常好的程序編輯器(或者IDE)來幫助你,你也得花了幾個小時才能發現這個問題所在。

3、   理解這些代碼將是非常難的事情。

你很難弄清楚一個全局變量是從哪里來得,它是用來做什么的。在開發的過程中,你可能會知道知道每一個全局變量,但大概一年之后,你可能會忘記其中至少一般的全局變量,這個時候你會為自己使用那么多全局變量而懊悔不已。

那么如果我們不使用全局變量,我們該使用什么呢?下面讓我們看看一些解決方案。

使用函數參數

停止使用全局變量的一種方法就是簡單的把變量作為函數的參數傳遞過去,如同下面所示:

$var='Hello World';test($var);

functiontest($var) {

echo$var;

}?>

如果你僅僅只需要傳遞一個全局變量,那么這是一種非常優秀甚至可以說是傑出的解決方案,但是如果你要傳遞很多個值,那該怎么辦呢?

比如說,假如我們要使用一個數據庫類,一個程序設置類和一個用戶類。在我們代碼中,這三個類在所有組件中都要用到,所以必須傳遞給每一個組件。如果我們使用函數參數的方法,我們不得不這樣:

$db= newDBConnection;$settings= newSettings_XML;$user= newUser;test($db,$settings,$user);

functiontest(&$db, &$settings, &$user) {// Do something}?>

顯然,這是不值得的,而且一旦我們有新的對象需要加入,我們不得不為每一個函數增加多一個函數參數。因此我們需要用采用另外一種方式來解決。

第一部分

使用單件(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;

ClassDBConnection{

var$user;

function &getInstance() {

static$me;

if (is_object($me) ==true) {

return$me;

}$me= newDBConnection;

return$me;

}

functionconnect() {// TODO}

functionquery() {// TODO}

}?>

上面例子中最重要的部分是函數getInstance()。這個函數通過使用一個靜態變量$me來返回這個類的實例,從而確保了只有一個DBConnection類的實例。

使用單件的好處就是我們不需要明確的傳遞一個對象,而是簡單的使用getInstance()方法來獲取到這個對象,就好像下面這樣:

然而使用單件也存在一系列的不足。首先,如果我們如何在一個類需要全局化多個對象呢?因為我們使用單件,所以這個不可能的(正如它的名字是單件一樣)。另 外一個問題,單件不能使用個體測試來測試的,而且這也是完全不可能的,除非你引入所有的堆棧,而這顯然是你不想看到的。這也是為什么單件不是我們理想中的 解決方法的主要原因。

注冊模式

讓一些對象能夠被我們代碼中所有的組件使用到(譯者注:全局化對象或者數據)的最好的方法就是使用一個中央容器對象,用它來包含我們所有的對象。通常這種容器對象被人們稱為一個注冊器。它非常的靈活而且也非常的簡單。一個簡單的注冊器對象就如下所示:

var$_objects= array();

functionset($name, &$object) {$this->_objects[$name] =&$object;

}

function &get($name) {

return$this->_objects[$name];

}

}?>

使用注冊器對象的第一步就是使用方法set()來注冊一個對象:

$db= newDBConnection;$settings= newSettings_XML;$user= newUser;// Register objects$registry=& newRegistry;$registry->set('db',$db);$registry->set('settings',$settings);$registry->set('user',$user);?>

現在我們的寄存器對象容納了我們所有的對象,我們指需要把這個注冊器對象傳遞給一個函數(而不是分別傳遞三個對象)。看下面的例子:

<?phpfunctiontest (&$registry) {$db=&$registry->get('db');$settings=&$registry->get('settings');$user=&$registry->get('user');// Do something with the objects}?>

注冊器相比其他的方法來說,它的一個很大的改進就是當我們需要在我們的代碼中新增加一個對象的時候,我們不再需要改變所有的東西(譯者注:指程序中所有 用到全局對象的代碼),我們只需要在注冊器里面新注冊一個對象,然后它(譯者注:新注冊的對象)就立即可以在所有的組件中調用。

為了更加容易的使用注冊器,我們把它的調用改成單件模式(譯者注:不使用前面提到的函數傳遞)。因為在我們的程序中只需要使用一個注冊器,所以單件模式使非常適合這種任務的。在注冊器類里面增加一個新的方法,如下所示:

static$me;

if (is_object($me) ==true) {

return$me;

}$me= newRegistry;

return$me;

}?>

這樣它就可以作為一個單件來使用,比如:

$db= newDBConnection;$settings= newSettings_XML;$user= newUser;// Register objects$registry=&Registry::getInstance();$registry->set('db',$db);$registry->set('settings',$settings);$registry->set('user',$user);

functiontest() {$registry=&Registry::getInstance();$db=&$registry->get('db');$settings=&$registry->get('settings');$user=&$registry->get('user');// Do something with the objects}?>

正如你看到的,我們不需要把私有的東西都傳遞到一個函數,也不需要使用“global”關鍵字。所以注冊器模式是這個問題的理想解決方案,而且它非常的靈活。

請求封裝器

雖然我們的注冊器已經使“global”關鍵字完全多余了,在我們的代碼中還是存在一種類型的全局變量:超級全局變量,比如變量$_POST,$ _GET。雖然這些變量都非常標准,而且在你使用中也不會出什么問題,但是在某些情況下,你可能同樣需要使用注冊器來封裝它們。

一個簡單的解決方法就是寫一個類來提供獲取這些變量的接口。這通常被稱為“請求封裝器”,下面是一個簡單的例子:

var$_request= array();

functionRequest() {// Get request variables$this->_request=$_REQUEST;

}

functionget($name) {

return$this->_request[$name];

}

}?>

上面的例子是一個簡單的演示,當然在請求封裝器(request wrapper)里面你還可以做很多其他的事情(比如:自動過濾數據,提供默認值等等)。

下面的代碼演示了如何調用一個請求封裝器:

$request= newRequest;// Register object$registry=&Registry::getInstance();$registry->set('request', &$request);test();

functiontest() {$registry=&Registry::getInstance();$request=&$registry->get('request');// Print the 'name' querystring, normally it'd be $_GET['name']echohtmlentities($request->get('name'));

}?>

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值