Forum und email

變數範圍

變數的範圍即它定義的上下文背景(也就是它的生效範圍)。大部分的 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 的例子:

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++;
}
?>

現在,每次調用 Test() 函式都會輸出 $a 的值並加一。

靜態變數也提供了一種處理遞歸函式的方法。遞歸函式是一種調用自己的函式。寫遞歸函式時要小心,因為可能會無窮遞歸下去。必須確保有充分的方法來中止遞歸。一下這個簡單的函式遞歸計數到 10,使用靜態變數 $count 來判斷何時停止:

Example#6 靜態變數與遞歸函式

<?php
function Test()
{
    static 
$count 0;

    
$count++;
    echo 
$count;
    if (
$count 10) {
        
Test();
    }
    
$count--;
}
?>

Note: 靜態變數可以按照上面的例子聲明。如果在聲明中用表達式的結果對其指派會導致解析錯誤。

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;
}
?>

全局和靜態變數的引用

在 Zend 引擎 1 代,它驅動了 PHP4,對於變數的 staticglobal 定義是以 references 的方式實現的。例如,在一個函式域內部用 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) {
}
    

類似的行為也適用於 static 語句。引用並不是靜態地存儲的:

<?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() 函式時其值並沒有被記住