變數的範圍即它定義的上下文背景(也就是它的生效範圍)。大部分的 PHP 變數只有一個單獨的範圍。這個單獨的範圍跨度同樣包含了 include 和 require 所引入的文件。例如:
<?php $a = 1; include 'b.inc'; ?>
這裡變數 $a 將會在包含檔案 b.inc 中生效。但是,在使用者自訂函數中,一個局部函數範圍將被引入。任何用於函數內部的變數按缺省情況將被限制在局部函數範圍內。例如:
<?php $a = 1; /* global scope */ function Test() { echo $a; /* reference to local scope variable */ } Test(); ?>
這個腳本不會有任何輸出,因為 echo 語句引用了一個局部版本的變數 $a,而且在這個範圍內,它並沒有被賦值。你可能注意到 PHP 的全域變數和 C 語言有一點點不同,在 C 語言中,全域變數在函數中自動生效,除非被局部變數覆寫。這可能引起一些問題,有些人可能不小心就改變了一個全域變數。 PHP 中全域變數在函數中使用時必須宣告為 global。
global 關鍵字
首先,一個使用 global 的例子:
Example #1 使用 global
<?php $a = 1; $b = 2; function Sum() { global $a, $b; $b = $a + $b; } Sum(); echo $b; ?>
以上腳本的輸出將是「3」。在函數中宣告了全域變數 $a 和 $b 之後,對任一變數的所有參考都會指向其全域版本。對於函數能夠宣告的全域變數的最大個數,PHP 沒有限制。
在全域範圍內存取變數的第二個辦法,是用特殊的 PHP 自訂 $GLOBALS 陣列。前面的例子可以寫成:
Example #2 使用 $GLOBALS 取代 global
<?php $a = 1; $b = 2; function Sum() { $GLOBALS['b'] = $GLOBALS['a'] + $GLOBALS['b']; } Sum(); echo $b; ?>
$GLOBALS 是一個關聯數組,每個變數為一個元素,鍵名對應變數名,值對應變數的內容。 $GLOBALS 之所以在全域範圍內存在,是因為 $GLOBALS 是一個超全域變數。以下範例顯示了超全域變數的用處:
Example #3 示範超全域變數和作用域的例子
<?php function test_global() { // 大多数的预定义变量并不 "super",它们需要用 'global' 关键字来使它们在函数的本地区域中有效。 global $HTTP_POST_VARS; echo $HTTP_POST_VARS['name']; // Superglobals 在任何范围内都有效,它们并不需要 'global' 声明。Superglobals 是在 PHP 4.1.0 引入的。 echo $_POST['name']; } ?>
使用靜態變數
變數範圍的另一個重要特性是靜態變數(static variable)。靜態變數僅在局部函數域中存在,但當程式執行離開此作用域時,其值並不會遺失。看看下面的範例:
Example #4 示範需要靜態變數的範例
<?php function Test() { $a = 0; echo $a; $a++; } ?>
本函數沒什麼用處,因為每次呼叫時都會將 $a 的值設為 0 並輸出 0。將變數加一的 $a++ 沒有作用,因為一旦退出本函數則變數 $a 就不存在了。要寫一個不會遺失本次計數值的計數函數,要將變數$a 定義為靜態的:
Example #5 使用靜態變數的例子
<?php function test() { static $a = 0; echo $a; $a++; } ?>
現在,變數$a 僅在第一次呼叫test( ) 函數時被初始化,之後每次呼叫test() 函數都會輸出$a 的值並加一。
靜態變數也提供了處理遞歸函數的方法。遞歸函數是一種呼叫自己的函數。寫遞歸函數時要小心,因為可能會無窮遞歸下去。必須確保有充分的方法來中止遞迴。以下這個簡單的函數遞迴計數到 10,使用靜態變數 $count 來判斷何時停止:
Example #6 靜態變數與遞歸函數
<?php function test() { static $count = 0; $count++; echo $count; if ($count < 10) { test(); } $count--; } ?>
靜態變數可以按照上面的範例宣告。如果在宣告中以表達式的結果對其賦值會導致解析錯誤。
Example #7 宣告靜態變數
<?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; } ?>
靜態宣告是在編譯時解析的。
在函數之外使用 global 關鍵字不算錯。可以用於在一個函數之內包含文件時。
全域和靜態變數的引用
在 Zend 引擎 1 代,它驅動了 PHP4,對於變數的 static 和 global 定義是以引用的方式實現的。例如,在一個函數域內部用 global 語句導入的一個真正的全域變數實際上是建立了一個到全域變數的參考。這有可能導致預料以外的行為,如以下範例所示範的:
<?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); ?>
以上程式會輸出:
NULL
object(stdClass)(0) {}
類似的語句行為也適用於staticic 。引用並不是靜態地儲存的:
<?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: NULL Static object: NULL Static object: NULL Static object: object(stdClass)(1) { ["property"]=>int(1) }
上例示範了當把一個引用賦值給一個靜態變數時,第二次呼叫&get_instance_ref() 函數時其值並沒有被記住。