溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

php變量作用域的詳細介紹

發(fā)布時間:2021-08-30 12:53:58 來源:億速云 閱讀:140 作者:chen 欄目:開發(fā)技術(shù)

本篇內(nèi)容介紹了“php變量作用域的詳細介紹”的有關(guān)知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細閱讀,能夠?qū)W有所成!

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

復(fù)制代碼 代碼如下:


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


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

復(fù)制代碼 代碼如下:


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

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

Test();
?> 

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

The global keyword
首先,一個使用 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,任何變量的所有引用變量都會指向到全局變量。對于一個函數(shù)能夠申明的全局變量的最大個數(shù),PHP 沒有限制。

在全局范圍內(nèi)訪問變量的第二個辦法,是用特殊的 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ù)組中,每一個變量為一個元素,鍵名對應(yīng)變量名,值變量的內(nèi)容。$GLOBALS 之所以在全局范圍內(nèi)存在,是因為 $GLOBALS 是一個超全局變量。以下范例顯示了超全局變量的用處:

例子 12-3. 演示超全局變量和作用域的例子

復(fù)制代碼 代碼如下:


<?php
function test_global()
{
   // 大多數(shù)的預(yù)定義變量并不 "super",它們需要用 'global' 關(guān)鍵字來使它們在函數(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)變量
變量范圍的另一個重要特性是靜態(tài)變量(static variable)。靜態(tài)變量僅在局部函數(shù)域中存在,但當(dāng)程序執(zhí)行離開此作用域時,其值并不丟失。看看下面的例子:

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

復(fù)制代碼 代碼如下:


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


本函數(shù)沒什么用處,因為每次調(diào)用時都會將 $a 的值設(shè)為 0 并輸出 "0"。將變量加一的 $a++ 沒有作用,因為一旦退出本函數(shù)則變量 $a 就不存在了。要寫一個不會丟失本次計數(shù)值的計數(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ù)都會輸出 $a 的值并加一。

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

例子 12-6. 靜態(tài)變量與遞歸函數(shù)

復(fù)制代碼 代碼如下:


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

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

注: 靜態(tài)變量可以按照上面的例子聲明。如果在聲明中用表達式的結(jié)果對其賦值會導(dǎo)致解析錯誤。

例子 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ū)動了 PHP4,對于變量的 static 和 global 定義是以 references 的方式實現(xiàn)的。例如,在一個函數(shù)域內(nèi)部用 global 語句導(dǎo)入的一個真正的全局變量實際上是建立了一個到全局變量的引用。這有可能導(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í)行以上例子會導(dǎo)致如下輸出:
NULLobject(stdClass)(0) {}
類似的行為也適用于 static 語句。引用并不是靜態(tài)地存儲的:

復(fù)制代碼 代碼如下:


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

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

function &get_instance_noref() {
   static $obj;

   echo "Static object: ";
   var_dump($obj);
   if (!isset($obj)) {
   // 將一個對象賦值給靜態(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í)行以上例子會導(dǎo)致如下輸出:
Static object: NULLStatic object: NULLStatic object: NULLStatic object: object(stdClass)(1) {  ["property"]=>  int(1)}

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

“php變量作用域的詳細介紹”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

php
AI