變量范圍

變量的范圍即它定義的上下文背景(也就是它的生效范圍)。大部分的 PHP 變量只有一個單獨的范圍。這個單獨的范圍跨度同樣包含了 include 和 require 引入的文件。例如:

<?php
$a 
1;
include 
'b.inc';
?>

這里變量 $a 將會在包含文件 b.inc 中生效。但是,在用戶自定義函數(shù)中,一個局部函數(shù)范圍將被引入。任何用于函數(shù)內(nèi)部的變量按缺省情況將被限制在局部函數(shù)范圍內(nèi)。例如:

<?php
$a 
1/* global scope */

function Test()
{
    echo 
$a/* reference to local scope variable */
}

Test();
?>

這個腳本不會有任何輸出,因為 echo 語句引用了一個局部版本的變量 $a,而且在這個范圍內(nèi),它并沒有被賦值。你可能注意到 PHP 的全局變量和 C 語言有一點點不同,在 C 語言中,全局變量在函數(shù)中自動生效,除非被局部變量覆蓋。這可能引起一些問題,有些人可能不小心就改變了一個全局變量。PHP 中全局變量在函數(shù)中使用時必須聲明為 global。

global 關(guān)鍵字

首先,一個使用 global 的例子:

示例 #1 使用 global

<?php
$a 
1;
$b 2;

function 
Sum()
{
    global 
$a$b;

    
$b $a $b;
}

Sum();
echo 
$b;
?>

以上腳本的輸出將是“3”。在函數(shù)中聲明了全局變量 $a$b 之后,對任一變量的所有引用都會指向其全局版本。對于一個函數(shù)能夠聲明的全局變量的最大個數(shù),PHP 沒有限制。

在全局范圍內(nèi)訪問變量的第二個辦法,是用特殊的 PHP 自定義 $GLOBALS 數(shù)組。前面的例子可以寫成:

示例 #2 使用 $GLOBALS 替代 global

<?php
$a 
1;
$b 2;

function 
Sum()
{
    
$GLOBALS['b'] = $GLOBALS['a'] + $GLOBALS['b'];
}

Sum();
echo 
$b;
?>

$GLOBALS 是一個關(guān)聯(lián)數(shù)組,每一個變量為一個元素,鍵名對應(yīng)變量名,值對應(yīng)變量的內(nèi)容。$GLOBALS 之所以在全局范圍內(nèi)存在,是因為 $GLOBALS 是一個超全局變量。以下范例顯示了超全局變量的用處:

示例 #3 演示超全局變量和作用域的例子

<?php
function test_superglobal()
{
    echo 
$_POST['name'];
}
?>

使用靜態(tài)變量

變量范圍的另一個重要特性是靜態(tài)變量(static variable)。靜態(tài)變量僅在局部函數(shù)域中存在,但當(dāng)程序執(zhí)行離開此作用域時,其值并不丟失??纯聪旅娴睦樱?

示例 #4 演示需要靜態(tài)變量的例子

<?php
function Test()
{
    
$a 0;
    echo 
$a;
    
$a++;
}
?>

本函數(shù)沒什么用處,因為每次調(diào)用時都會將 $a 的值設(shè)為 0 并輸出 0。將變量加一的 $a++ 沒有作用,因為一旦退出本函數(shù)則變量 $a 就不存在了。要寫一個不會丟失本次計數(shù)值的計數(shù)函數(shù),要將變量 $a 定義為靜態(tài)的:

示例 #5 使用靜態(tài)變量的例子

<?php
function test()
{
    static 
$a 0;
    echo 
$a;
    
$a++;
}
?>

現(xiàn)在,變量 $a 僅在第一次調(diào)用 test() 函數(shù)時被初始化,之后每次調(diào)用 test() 函數(shù)都會輸出 $a 的值并加一。

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

示例 #6 靜態(tài)變量與遞歸函數(shù)

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

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

常量表達(dá)式的結(jié)果可以賦值給靜態(tài)變量,但是動態(tài)表達(dá)式(比如函數(shù)調(diào)用)會導(dǎo)致解析錯誤。

示例 #7 聲明靜態(tài)變量

<?php
function foo(){
    static 
$int 0;          // correct
    
static $int 1+2;        // correct 
    
static $int sqrt(121);  // wrong  (as it is a function)

    
$int++;
    echo 
$int;
}
?>

靜態(tài)聲明是在編譯時解析的。

全局和靜態(tài)變量的引用

對于變量的 staticglobal 定義是以引用的方式實現(xiàn)的。例如,在一個函數(shù)域內(nèi)部用 global 語句導(dǎo)入的一個真正的全局變量實際上是建立了一個到全局變量的引用。這有可能導(dǎo)致預(yù)料之外的行為,如以下例子所演示的:

<?php
function test_global_ref() {
    global 
$obj;
    
$new = new stdclass;
    
$obj = &$new;
}

function 
test_global_noref() {
    global 
$obj;
    
$new = new stdclass;
    
$obj $new;
}

test_global_ref();
var_dump($obj);
test_global_noref();
var_dump($obj);
?>

以上例程會輸出:

NULL
object(stdClass)#1 (0) {
}

類似的行為也適用于 static 語句。引用并不是靜態(tài)地存儲的:

<?php
function &get_instance_ref() {
    static 
$obj;

    echo 
'Static object: ';
    
var_dump($obj);
    if (!isset(
$obj)) {
        
$new = new stdclass;
        
// 將一個引用賦值給靜態(tài)變量
        
$obj = &$new;
    }
    if (!isset(
$obj->property)) {
        
$obj->property 1;
    } else {
        
$obj->property++;
    }
    return 
$obj;
}

function &
get_instance_noref() {
    static 
$obj;

    echo 
'Static object: ';
    
var_dump($obj);
    if (!isset(
$obj)) {
        
$new = new stdclass;
        
// 將一個對象賦值給靜態(tài)變量
        
$obj $new;
    }
    if (!isset(
$obj->property)) {
        
$obj->property 1;
    } else {
        
$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)#3 (1) {
  ["property"]=>
  int(1)
}

上例演示了當(dāng)把一個引用賦值給一個靜態(tài)變量時,第二次調(diào)用 &get_instance_ref() 函數(shù)時其值并沒有被記住。