PHP之引用
??? 所謂PHP的引用,就是不同的名字訪問同一個變數內容。可用在變數、函數以及對象上,用法就是在它們前面加上&符號。下面來細說下引用的類型及作用:
?
一、參考型別
1.1、變數引用:兩個變數指向同一個內容
1.2、函數傳址引用
function test(&$a){$a=$a+100; }$b=1;test($b); //這裡$b傳遞給函數的其實是$b的變數內容所處的記憶體位址,通過在函數裡改變$a的值,就可以改變$b的值了 echo $b;//輸出101 ?
要注意的是,在這裡test(1); 的話就會出錯,原因自己去想
1.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而已, 而$a做任何改變都不會影響到函數中的$b, 而通過$a=&test()方式調用函數呢, 他的作用是將return $b中的$b變數的記憶體位址與$a變數的記憶體位址 指向了同一個地方, 即產生了相當於這樣的效果($a=&b;) 所以改變$a的值, 也同時改變了$b的值, 所以在執行了
$a=&test();
$a=5;
以後,$b的值變為了5
這裡是為了方便理解函數的引用返回才使用靜態變數,其實函數的引用返回多用在對象中
?
1.4、對象的引用
class testa{var $abc="ABC"; }$b=new testa;$c = &$b; // or $c = $b;echo $b->abc;//這裡輸出ABC echo $c->abc;//這裡輸出ABC $b->abc="DEF"; echo $c->abc;//這裡輸出DEF ?
以上代碼是在PHP5中的運行效果;
在PHP5中,對象的複製 是通過引用來實現的。上例中$b=new a; $c=$b; 其實等效於$b=new a; $c=&$b;
PHP5中預設就是通過引用來調用對象, 但有時你可能想建立一個對象的副本,並希望原來的對象的改變不影響到副本。 為了這樣的目的,PHP定義了一個特殊的方法,稱為__clone.
二、引用的作用
如果程式比較大,引用同一個對象的變數比較多,並且希望用完該對象後手工清除它,個人建議用 "&" 方式,然後用$var=null的方式清除。其它時候還是用php5的預設吧。另外, php5中對於大數組的傳遞,建議用 "&" 方式, 畢竟節省記憶體空間使用。
三、取值 (Dereference)
當你 unset 一個引用,只是斷開了變數名和變數內容之間的綁定。這並不意味著變數內容被銷毀了。例如:
?? $a = 1;
?? $b =& $a;
?? unset ($a);
?>
不會 unset $b,只是銷毀了 $a。
再如global 引用,當用 global $var 聲明一個變數時實際上建立了一個到全域變數的引用。也就是說和這樣做是相同的:
??? $var =& $GLOBALS["var"];
?>
這意味著,例如,unset $var 不會 unset 全域變數。
$this 的引用:在一個對象的方法中,$this 永遠是調用它的對象的引用。
?
PS:
PHP有個好處就是可利用其擁有的自動記憶體回收機制(釋放記憶體)。即不需要在使用完變數後做任何釋放記憶體的處理,PHP會幫你完成。當然,我們可以按自己的意願調用 unset() 函數來釋放記憶體,但通常不需要這麼做。
不過在PHP裡,至少有一種情況記憶體不會得到自動釋放,如下面:
如果兩個對象之間存在著相互引用的關係,如“父物件-子物件”,對父物件調用 unset()不會釋放在子物件中引用父物件的記憶體(即便父物件被記憶體回收,也不行)。
這個時候,即便是手動調用 unset()。詳情可考:http://bugs.php.net/bug.php?id=33595。
折中的辦法是在 unset() 函數中調用對象中的 __destruct() 方法;
?
?
------------------------------------------------
補充:
php中對於地址的指向(類似指標)功能不是由使用者自己來實現的,是由Zend核心實現的,php中引用採用的是“寫時拷貝”的原理,就是除非發生寫操作,指向同一個地址的變數或者對象是不會被拷貝的。
通俗的講
1:如果有下面的代碼
$a="ABC";
$b=$a;
其實此時,$a與$b都是指向同一記憶體位址,而並不是$a與$b佔用不同的記憶體
2:如果在上面的代碼基礎上再加上如下代碼
$a="EFG";
由於$a與$b所指向的記憶體的資料要重新寫一次了,此時Zend核心會自動判斷,自動為$b生產一個$a的資料拷貝,重新申請一塊記憶體進行儲存。