|
昨晚就與到這么一個(gè)問(wèn)題,是全局變量在函數(shù)中的問(wèn)題。今天搜索了一下,發(fā)現(xiàn)一篇相當(dāng)不錯(cuò)的文章,講了php中的變量作用域。是一位網(wǎng)友翻譯的在這貼一下:
變量范圍
變量的范圍即它定義的上下文背景(譯者:說(shuō)白了,也就是它的生效范圍)。大部分的 php 變量只有一個(gè)單獨(dú)的范圍。這個(gè)單獨(dú)的范圍跨度同樣包含了 include 和 require 引入的文件。范例:
復(fù)制代碼 代碼如下:
<?php
$a = 1;
include "b.inc";
?>
這里變量 $a 將會(huì)在包含文件 b.inc 中生效。但是,在用戶自定義函數(shù)中,一個(gè)局部函數(shù)范圍將被引入。任何用于函數(shù)內(nèi)部的變量按缺省情況將被限制在局部函數(shù)范圍內(nèi)。范例:
復(fù)制代碼 代碼如下:
<?php
$a = 1; /* global scope */
function Test()
{
echo $a; /* reference to local scope variable */
}
Test();
?>
這個(gè)腳本不會(huì)有任何輸出,因?yàn)?echo 語(yǔ)句引用了一個(gè)局部版本的變量 $a,而且在這個(gè)范圍內(nèi),它并沒(méi)有被賦值。你可能注意到 php 的全局變量和 C 語(yǔ)言有一點(diǎn)點(diǎn)不同,在 C 語(yǔ)言中,全局變量在函數(shù)中自動(dòng)生效,除非被局部變量覆蓋。這可能引起一些問(wèn)題,有些人可能漫不經(jīng)心的改變一個(gè)全局變量。php 中全局變量在函數(shù)中使用時(shí)必須申明為全局。
The global keyword
首先,一個(gè)使用 global 的例子:
例子 12-1. 使用 global
復(fù)制代碼 代碼如下:
<?php
$a = 1;
$b = 2;
function Sum()
{
global $a, $b;
$b = $a + $b;
}
Sum();
echo $b;
?>
以上腳本的輸出將是 "3"。在函數(shù)中申明了全局變量 $a 和 $b,任何變量的所有引用變量都會(huì)指向到全局變量。對(duì)于一個(gè)函數(shù)能夠申明的全局變量的最大個(gè)數(shù),php 沒(méi)有限制。
在全局范圍內(nèi)訪問(wèn)變量的第二個(gè)辦法,是用特殊的 php 自定義 $GLOBALS 數(shù)組。前面的例子可以寫成:
例子 12-2. 使用 $GLOBALS 替代 global
復(fù)制代碼 代碼如下:
<?php
$a = 1;
$b = 2;
function Sum()
{
$GLOBALS["b"] = $GLOBALS["a"] + $GLOBALS["b"];
}
Sum();
echo $b;
?>
在 $GLOBALS 數(shù)組中,每一個(gè)變量為一個(gè)元素,鍵名對(duì)應(yīng)變量名,值變量的內(nèi)容。$GLOBALS 之所以在全局范圍內(nèi)存在,是因?yàn)?$GLOBALS 是一個(gè)超全局變量。以下范例顯示了超全局變量的用處:
例子 12-3. 演示超全局變量和作用域的例子
復(fù)制代碼 代碼如下:
<?php
function test_global()
{
// 大多數(shù)的預(yù)定義變量并不 "super",它們需要用 'global' 關(guān)鍵字來(lái)使它們?cè)诤瘮?shù)的本地區(qū)域中有效。
global $HTTP_POST_VARS;
print $HTTP_POST_VARS['name'];
// Superglobals 在任何范圍內(nèi)都有效,它們并不需要 'global' 聲明。Superglobals 是在 php 4.1.0 引入的。
print $_POST['name'];
}
?>
使用靜態(tài)變量
變量范圍的另一個(gè)重要特性是靜態(tài)變量(static variable)。靜態(tài)變量?jī)H在局部函數(shù)域中存在,但當(dāng)程序執(zhí)行離開此作用域時(shí),其值并不丟失。看看下面的例子:
例子 12-4. 演示需要靜態(tài)變量的例子
復(fù)制代碼 代碼如下:
<?php
function Test ()
{
$a = 0;
echo $a;
$a++;
}
?>
本函數(shù)沒(méi)什么用處,因?yàn)槊看握{(diào)用時(shí)都會(huì)將 $a 的值設(shè)為 0 并輸出 "0"。將變量加一的 $a++ 沒(méi)有作用,因?yàn)橐坏┩顺霰竞瘮?shù)則變量 $a 就不存在了。要寫一個(gè)不會(huì)丟失本次計(jì)數(shù)值的計(jì)數(shù)函數(shù),要將變量 $a 定義為靜態(tài)的:
例子 12-5. 使用靜態(tài)變量的例子
復(fù)制代碼 代碼如下:
<?php
function Test()
{
static $a = 0;
echo $a;
$a++;
}
?>
現(xiàn)在,每次調(diào)用 Test() 函數(shù)都會(huì)輸出 $a 的值并加一。
靜態(tài)變量也提供了一種處理遞歸函數(shù)的方法。遞歸函數(shù)是一種調(diào)用自己的函數(shù)。寫遞歸函數(shù)時(shí)要小心,因?yàn)榭赡軙?huì)無(wú)窮遞歸下去。必須確保有充分的方法來(lái)中止遞歸。一下這個(gè)簡(jiǎn)單的函數(shù)遞歸計(jì)數(shù)到 10,使用靜態(tài)變量 $count 來(lái)判斷何時(shí)停止:
例子 12-6. 靜態(tài)變量與遞歸函數(shù)
復(fù)制代碼 代碼如下:
<?php
function Test()
{
static $count = 0;
$count++;
echo $count;
if ($count < 10) {
Test ();
}
$count--;
}
?>
注: 靜態(tài)變量可以按照上面的例子聲明。如果在聲明中用表達(dá)式的結(jié)果對(duì)其賦值會(huì)導(dǎo)致解析錯(cuò)誤。
例子 12-7. 聲明靜態(tài)變量
復(fù)制代碼 代碼如下:
<?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;
}
?>
全局和靜態(tài)變量的引用
在 Zend 引擎 1 代,驅(qū)動(dòng)了 php4,對(duì)于變量的 static 和 global 定義是以 references 的方式實(shí)現(xiàn)的。例如,在一個(gè)函數(shù)域內(nèi)部用 global 語(yǔ)句導(dǎo)入的一個(gè)真正的全局變量實(shí)際上是建立了一個(gè)到全局變量的引用。這有可能導(dǎo)致預(yù)料之外的行為,如以下例子所演示的:
復(fù)制代碼 代碼如下:
<?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);
?>
執(zhí)行以上例子會(huì)導(dǎo)致如下輸出:
NULLobject(stdClass)(0) {}
類似的行為也適用于 static 語(yǔ)句。引用并不是靜態(tài)地存儲(chǔ)的:
復(fù)制代碼 代碼如下:
<?php
function &get_instance_ref() {
static $obj;
echo "Static object: ";
var_dump($obj);
if (!isset($obj)) {
// 將一個(gè)引用賦值給靜態(tài)變量
$obj = &new stdclass;
}
$obj->property++;
return $obj;
}
function &get_instance_noref() {
static $obj;
echo "Static object: ";
var_dump($obj);
if (!isset($obj)) {
// 將一個(gè)對(duì)象賦值給靜態(tài)變量
$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();
?>
執(zhí)行以上例子會(huì)導(dǎo)致如下輸出:
Static object: NULLStatic object: NULLStatic object: NULLStatic object: object(stdClass)(1) { ["property"]=> int(1)}
上例演示了當(dāng)把一個(gè)引用賦值給一個(gè)靜態(tài)變量時(shí),第二次調(diào)用 &get_instance_ref() 函數(shù)時(shí)其值并沒(méi)有被記住。
php技術(shù):淺析php變量作用域的一些問(wèn)題,轉(zhuǎn)載需保留來(lái)源!
鄭重聲明:本文版權(quán)歸原作者所有,轉(zhuǎn)載文章僅為傳播更多信息之目的,如作者信息標(biāo)記有誤,請(qǐng)第一時(shí)間聯(lián)系我們修改或刪除,多謝。