PHP 中的每個變數都有一個針對它的範圍,它是指可以在其中訪問變數(從而訪問它的值)的一個領域。對於初學者來說,變數的範圍是它們所駐留的頁面。因此,如果你定義了 $var,頁面餘下部分就可以訪問 $var,但是,其它頁面一般不能訪問它(除非使用特殊的變數)。
因為包含檔案像它們是原始(包含)指令碼的一部分那樣工作,所以在 include() 那一行之前定義的變數可供包含檔案使用。此外,包含檔案內定義的變數可供 include() 那一行之後的父(包含)指令碼使用。
當使用你自己定義的函數時,所有這些都將變得不那麼明顯。這些函數具有它們自己的範圍,這意味著在一個函數內使用的變數不能在其外部使用,在一個函數外部定義的變數不能在其內部使用。由於這個原因,函數內部的變數可以具有與其外部的變數相同的名稱,但是它們仍然是完全不同的變數,並且具有不同的值。對於大多數初級程式員來說,這是一個使人糊塗的概念。
要改變一個函數內的變數的範圍,可以使用 global 語句。
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
function function_name() {
global $var;
}
$var=20;
function_name(); // Function call.
?>
在這個樣本中,函數內部的 $var 現在與函數外部的 $var 相同。這意味著變數 $var 已經具有一個值20,如果在函數內部改變了這個值,外部的 $var 值也會改變。
避開變數範圍的另一個方法是利用超全域變數:$_GET、$_POST、$_REQUEST 等。這些變數在你的函數內是自動可訪問的(因此,它們是超全域變數)。也可以添加元素到 $GLOBALS 數組中,使得可以在函數內使用它們。
也就是說,最好不要在函數內使用全域變數。在設計函數時,應該使它們根據需要接受每個值作為參數,並根據需要返回任何值。依靠函數內的全域變數將使得它們更依賴於上下文,因而不太有用。
在PHP中變數主要有:內建超級全域變數,一般的變數,常量,全域變數,靜態變數等.
內建超級全域變數可以在指令碼的任何地方使用和可見。即如果我們在一個PHP頁面中改變了其中的一個值,那麼在其他PHP頁面中使用時,它的值也會發生改變。
•常量一旦被聲明將可以在全域可見,也就是說,它們可以函數內外使用,但是這僅僅限於一個頁面之中(包含我們通過include和include_once)包含進來的PHP指令碼,但是在其他的頁面中就不能使用了。
•在一個指令碼中聲明的全域變數在整個指令碼中是可見的,但不是在函數內部,在函數內部的變數如果與全域變數名稱相同,以函數內部的變數為準。
•函數內部使用的變數聲明為全域變數時,其名稱要與全域變數的名稱一致,在這樣的情況下,我們就可以在函數中使用函數外部的全域變數了,這樣就可以避免上一種因為函數內部的變數與外部的全域變數名稱相同而覆蓋了外部變數這樣的情況。
•在函數內部建立並聲明為靜態變數無法在函數外部可見,但是可以在函數的多次執行過程中保持該值,最常見的情況就是在函數的遞迴執行的過程之中。
•在函數內部建立的變數對函數來說是本地的,而當函數終止時,該變數也就不存在了。
超級全域變數的完整列表如下:
•.$GOBALS 所有全域變數數組
•.$_SERVER 伺服器環境變數數組
•.$_POST 通過POST方法傳遞給該指令碼的變數數組
•.$_GET 通過GET方法傳遞給該指令碼的變數數組
•.$_COOKIE cookie變數數組
•.$_FILES 與檔案上傳相關的變數數組
•.$_ENV 環境變數數組
•.$_REQUEST 所有使用者輸入的變數數組包括$_GET $_POST $_COOKIE 所包含的輸入內容
•.$_SESSION 會話變數數組
執行個體講解:
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
$a = 4;
function sendValue($x)
{
echo $x;
}
sendValue($a);
?>
講解: $a定義在函數外,函數定義了參數,當函數被調用時,$a將以參數的形式被傳遞。因此上面代碼能夠正常運行。
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
$a = 4;
function sendValue()
{
echo $a;
}
sendValue();
?>
講解:當函數被調用時,$a不能以參數的形式被傳遞。所以上面代碼不能夠正常運行。
變數範圍
變數的範圍即它定義的上下文背景(譯者:說白了,也就是它的生效範圍)。大部分的 PHP 變數只有一個單獨的範圍。這個單獨的範圍跨度同樣包含了 include 和 require 引入的檔案。範例:
複製代碼 代碼如下:
<?php
$a = 1;
include "b.inc";
?>
這裡變數 $a 將會在包含檔案 b.inc 中生效。但是,在使用者自訂函數中,一個局部函數範圍將被引入。任何用於函數內部的變數按預設情況將被限制在局部函數範圍內。範例:
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
$a = 1; /* global scope */
function Test()
{
echo $a; /* reference to local scope variable */
}
Test();
?>
這個指令碼不會有任何輸出,因為 echo 語句引用了一個局部版本的變數 $a,而且在這個範圍內,它並沒有被賦值。你可能注意到 PHP 的全域變數和 C 語言有一點點不同,在 C 語言中,全域變數在函數中自動生效,除非被局部變數覆蓋。這可能引起一些問題,有些人可能漫不經心的改變一個全域變數。PHP 中全域變數在函數中使用時必須申明為全域。
The global keyword
首先,一個使用 global 的例子:
例子 12-1. 使用 global
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
$a = 1;
$b = 2;
function Sum()
{
global $a, $b;
$b = $a + $b;
}
Sum();
echo $b;
?>
以上指令碼的輸出將是 "3"。在函數中申明了全域變數 $a 和 $b,任何變數的所有引用變數都會指向到全域變數。對於一個函數能夠申明的全域變數的最大個數,PHP 沒有限制。
在全域範圍內訪問變數的第二個辦法,是用特殊的 PHP 自訂 $GLOBALS 數組。前面的例子可以寫成:
例子 12-2. 使用 $GLOBALS 替代 global
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
$a = 1;
$b = 2;
function Sum()
{
$GLOBALS["b"] = $GLOBALS["a"] + $GLOBALS["b"];
}
Sum();
echo $b;
?>
在 $GLOBALS 數組中,每一個變數為一個元素,鍵名對應變數名,值變數的內容。$GLOBALS 之所以在全域範圍記憶體在,是因為 $GLOBALS 是一個超全域變數。以下範例顯示了超全域變數的用處:
例子 12-3. 示範超全域變數和範圍的例子
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
function test_global()
{
// 大多數的預定義變數並不 "super",它們需要用 'global' 關鍵字來使它們在函數的本地地區中有效。
global $HTTP_POST_VARS;
print $HTTP_POST_VARS['name'];
// Superglobals 在任何範圍內都有效,它們並不需要 'global' 聲明。Superglobals 是在 PHP 4.1.0 引入的。
print $_POST['name'];
}
?>
使用靜態變數
變數範圍的另一個重要特性是靜態變數(static variable)。靜態變數僅在局部函數域中存在,但當程式執行離開此範圍時,其值並不丟失。看看下面的例子:
例子 12-4. 示範需要靜態變數的例子
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
function Test ()
{
$a = 0;
echo $a;
$a++;
}
?>
本函數沒什麼用處,因為每次調用時都會將 $a 的值設為 0 並輸出 "0"。將變數加一的 $a++ 沒有作用,因為一旦退出本函數則變數 $a 就不存在了。要寫一個不會丟失本次計數值的計數函數,要將變數 $a 定義為靜態:
例子 12-5. 使用靜態變數的例子
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
function Test()
{
static $a = 0;
echo $a;
$a++;
}
?>
現在,每次調用 Test() 函數都會輸出 $a 的值並加一。
靜態變數也提供了一種處理遞迴函式的方法。遞迴函式是一種調用自己的函數。寫遞迴函式時要小心,因為可能會無窮遞迴下去。必須確保有充分的方法來中止遞迴。一下這個簡單的函數遞迴計數到 10,使用靜態變數 $count 來判斷何時停止:
例子 12-6. 靜態變數與遞迴函式
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
function Test()
{
static $count = 0;
$count++;
echo $count;
if ($count < 10) {
Test ();
}
$count--;
}
?>
注: 靜態變數可以按照上面的例子聲明。如果在聲明中用運算式的結果對其賦值會導致解析錯誤。
例子 12-7. 聲明靜態變數
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
function foo(){
static $int = 0; // correct
static $int = 1+2; // wrong (as it is an expression)
static $int = sqrt(121); // wrong (as it is an expression too)
$int++;
echo $int;
}
?>
全域和靜態變數的引用
在 Zend 引擎 1 代,驅動了 PHP4,對於變數的 static 和 global 定義是以 references 的方式實現的。例如,在一個函數域內部用 global 語句匯入的一個真正的全域變數實際上是建立了一個到全域變數的引用。這有可能導致預料之外的行為,如以下例子所示範的:
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
function test_global_ref() {
global $obj;
$obj = &new stdclass;
}
function test_global_noref() {
global $obj;
$obj = new stdclass;
}
test_global_ref();
var_dump($obj);
test_global_noref();
var_dump($obj);
?>
執行以上例子會導致如下輸出:
複製代碼 代碼如下:
NULLobject(stdClass)(0) {}
類似的行為也適用於 static 語句。引用並不是靜態地儲存的:
複製代碼 代碼如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php
function &get_instance_ref() {
static $obj;
echo "Static object: ";
var_dump($obj);
if (!isset($obj)) {
// 將一個引用賦值給靜態變數
$obj = &new stdclass;
}
$obj->property++;
return $obj;
}
function &get_instance_noref() {
static $obj;
echo "Static object: ";
var_dump($obj);
if (!isset($obj)) {
// 將一個對象賦值給靜態變數
$obj = new stdclass;
}
$obj->property++;
return $obj;
}
$obj1 = get_instance_ref();
$still_obj1 = get_instance_ref();
echo "\n";
$obj2 = get_instance_noref();
$still_obj2 = get_instance_noref();
?>
執行以上例子會導致如下輸出:
複製代碼 代碼如下:
Static object: NULLStatic object: NULLStatic object: NULLStatic object: object(stdClass)(1) { ["property"]=> int(1)}
上例示範了當把一個引用賦值給一個靜態變數時,第二次調用 &get_instance_ref() 函數時其值並沒有被記住。