天天看點

位址引用PHP,淺談PHP變量作用域以及位址引用問題

作用域的概念:

在PHP腳本的任何位置都可以聲明變量,但是,聲明變量的位置會大大影響通路變量的範圍。這個可以通路的範圍稱為作用域。

主要的常用的包括:局部變量、全局變量、靜态變量。

1、局部變量:就是在函數内聲明的變量,他儲存在記憶體的棧内,是以通路速度很快。僅在函數内有效。

2、全局變量:與局部變量相反,全局變量可以在程式的任何地方通路。隻要在變量前面加關鍵字GLOBAL,就可以将其識别為全局變量。在整個php檔案内有效。

3、靜态變量:用static修飾隻存在于函數作用域的變量,函數執行結束後其值并不消失。注:初始化後不能再次進行初始化,不能用表達式來指派。

function test()

{

static $b=0;//申明靜态變量,放在函數外部聲明的話,在函數内部是用不到的

$b=$b+1;

echo $b;

}

test();//這條語句會輸出 $b的值 為1

test();//這條語句會輸出 $b的值 為2

注:static $b=0 這一指派操作隻會在變量第一次被初始化的時候執行。

附A:類中靜态成員和靜态方法,差不多隻是調用的時候統一使用類名或者self或者parent加::xxx,他們的作用域和這個一樣,但是他的聲明是在方法外部的

附B:js裡面的作用域使:用var aa=‘xxx';在函數外面聲明的就是全局變量(不管是否帶有修飾符var)。在函數内部使用 var聲明的是局部變量,不使用var修飾的是全局變量。

附C:關于引用

PHP引用:就是在變量、函數或者對象前加&.php中的引用就是想用不同的名字通路同一個變量的内容。

1、變量的引用:

$a="ABC";

$b =&$a;

echo $a;//這裡輸出:ABC

echo $b;//這裡輸出:ABC

$b="EFG";

echo $a;//這裡$a的值變為EFG 是以輸出EFG

echo $b;//這裡輸出EFG

2、函數的傳址調用

function test(&$a)

{

$a=$a+100;

}

$b=1;

echo $b;//輸出1

test($b); //這裡$b傳遞給函數的其實是$b的變量内容所處的記憶體位址,通過在函數裡改變$a的值 就可以改變$b的值了

echo "

";

echo $b;//輸出101

3、函數的引用傳回

function &test()

{

static $b=0;//申明一個靜态變量

$b=$b+1;

echo $b;

return $b;

}

$a=test();//這條語句會輸出 $b的值 為1

$a=5;

$a=test();//這條語句會輸出 $b的值 為2

$a=&test();//這條語句會輸出 $b的值 為3

$a=5;

$a=test();//這條語句會輸出 $b的值 為6

解析:使用$a=test()得到的其實不是函數的引用傳回。隻是将函數的傳回值複制給$a,而不會影響到$b。這樣調用和普通的調用沒差別。

Php規定:$a=&test()方式得到才是函數的引用傳回。他将$b變量的記憶體位址和$a變量的記憶體位址指向了同一個地方。即相當于$a=&$b;

4、取消引用

$a = 1;

$b =& $a;

unset ($a);

echo $b;

解析:unset一個引用,隻是取消了變量名和變量的内容之間的綁定,并不意味着内容被銷毀,其值還是真實存在的。

5、global引用:使用global $var  聲明一個變量時,其實就是建立了一個到全局變量的引用。Global $val <=> $var=&$GLOBALS[‘var'] ;

6、對象的引用:在對象的方法中,$this調用的對象都是調用它的引用

注:php中對于位址的指向不是由使用者自己來實作的,而是通過zend核心實作的,php的引用采用的是“寫拷貝”的原理,就是除非發生寫操作,指向同一個位址的變量或對象是不會被拷貝的。

$a = 1;

$b =$a;

$a和$b都是指向同一個記憶體位址,并不是$a和$b占用不同的記憶體。

若是現在執行一句$a=”dsd”:$a和$b所指向的記憶體資料需要重新寫一次,此時zend核心會自動判斷。自動為$b産生一個$a的資料拷貝,重新申請一塊記憶體進行存儲。