PHP語言中global和$GLOBALS[]的分析
php中global和$GLOBALS不僅僅是寫法不一樣以為,2者的區別還是很大的,在實際應用中需要注意!
先看下面的例子:
引用
執行結果為:
0
5
怎麼會這樣呢?不應該是2個5嗎?怎麼會出現1個0和1個5呢?
恩,我們保留以上問題,深入分析$GLOBALS和global的原理!
我們都知道變數其實是相應實體記憶體在代碼中的”代號”而已
引用php手冊的$GLOBALS的解釋:
引用
Global 變數:$GLOBALS,注意: $GLOBALS 在 PHP 3.0.0 及以後版本中適用。
由所有已定義全域變數組成的數組。變數名就是該數組的索引。這是一個“superglobal”,或者可以描述為自動全域變數。
也就是說上面代碼中的$var1和$GLOBALS['var1']是指的同一變數,而不是2個不同的變數!
下面來分析global到底做了什嗎?
引用php手冊的global的解釋:
引用
如果在一個函數內部給一個聲明為 global 的變數賦於一個引用,該引用只在函數內部可見。可以通過使用 $GLOBALS 數組避免這一點。
我們都知道php中的函數所產生的變數都是函數的私人變數,那麼global關鍵字產生的變數也肯定逃不出這個規則,為什麼這麼說呢,看下面的代碼:
引用
執行結果為:
1
為什麼會輸出1呢?不是已經把$a給unset了嗎?unset失靈了?php的bug?
都不是,其實unset起作用了,是把test函數中的$a給unset掉了,可以在函數test()中加入
print $a;
來測試!
接著回到上面的例子1,看test_global中的這一代碼“$var2 =& $var1;”,上面是一個引用賦值運算,也就是$var2將指向var1所指向的實體記憶體地址,所以例子1執行過test_global函數以後,變數的變化只在函數的局部產生效應,在函數外部$var2的指向實體記憶體地址並沒有變化,還是它自己.
此時,就能理解為什麼例子1執行完以後,$var2是0,而$var3是5了!
所以我們得出一個結論,在函數中global和$GLOBALS[]的區別在於:
global在函數產生一個指向函數外部變數的別名變數,而不是真正的函數外部變數,一但改變了別名變數的指向地址,就會發生一些意料不到情況,例如例子 1.
$GLOBALS[]確確實實調用是外部的變數,函數內外會始終保持一致
可以對照 下面兩個列子再加深下印象:
global:
輸出:someting
$GLOBALS[]:
輸出:空
當按照上面的思路理解後,碰到下面的情況是不是又有些暈呢?
輸出將是 “3″。在函數中申明 了全域變數 $a 和 $b,任何變數的所有引用變數都會指向到全域變數。
怎麼不是2呢,在函數外部不是不影響嗎,請注意$b在函數中並沒有通過引用修改,而是修改的$b指向實體記憶體的值,因此外部輸入為3。