PHP取地址符(引用符)&詳解

pythontab發表於2018-01-25

php的引用就是在變數或者函式、物件等前面加上&符號。在PHP 中引用的意思是:不同的名字訪問同一個變數內容。與C語言中的指標是有差別的,C語言中的指標裡面儲存的是變數的內容在記憶體中存放的地址。

變數的引用

PHP 的引用允許你用兩個變數來指向同一個內容。

<? 
$a="ABC"; 
$b =&$a; 
echo $a;//這裡輸出:ABC 
echo $b;//這裡輸出:ABC 
$b="EFG"; 
echo $a;//這裡$a的值變為EFG 所以輸出EFG 
echo $b;//這裡輸出EFG

函式的傳址呼叫

傳址呼叫我就不多說了,下面直接給出程式碼:

function test(&$a){
    $a=$a+100;
}
$b=1;
echo $b;//輸出1
test($b);   //這裡$b傳遞給函式的其實是$b的變數內容所處的記憶體地址,透過在函式里改變$a的值 就可以改變$b的值了
echo "<br />";
echo $b;//輸出101

要注意的是,在這裡test(1);的話就會出錯,原因自己去想。

函式的引用返回

先看程式碼:

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();得到的其實不是函式的引用返回,這跟普通的函式呼叫沒有區別。至於原因:這是PHP的規定。PHP規定透過$a=&test(); 方式得到的才是函式的引用返回。至於什麼是引用返回呢(PHP手冊上說:引用返回用在當想用函式找到引用應該被繫結在哪一個變數上面時。)

用上面的例子來解釋就是:

$a=test()方式呼叫函式,只是將函式的值賦給$a而已,而$a做任何改變都不會影響到函式中的$b。而透過$a=&test()方式呼叫函式呢,他的作用是將return $b中的 $b變數的記憶體地址與$a變數的記憶體地址 指向了同一個地方即產生了相當於這樣的效果($a=&b;) 所以改變$a的值也同時改變了$b的值,所以在執行了

$a=&test();
$a=5;

以後,$b的值變為了5。

這裡是為了讓大家理解函式的引用返回才使用靜態變數的,其實函式的引用返回多用在物件中。

物件的引用

<?
class a{
    var $abc="ABC";
}
$b=new a;
$c=$b;
echo $b->abc;//這裡輸出ABC
echo $c->abc;//這裡輸出ABC
$b->abc="DEF";
echo $c->abc;//這裡輸出DEF
?>

在PHP5中物件的複製是透過引用來實現的。上列中$b=new a; $c=$b; 其實等效於$b=new a; $c=&$b;PHP5中預設就是透過引用來呼叫物件,但有時你可能想建立一個物件的副本,並希望原來的物件的改變不影響到副本。為了這樣的目的,PHP定義了一個特殊的方法,稱為__clone.

引用的作用:如果程式比較大,引用同一個物件的變數比較多,並且希望用完該物件後手工清除它,個人建議用 "&" 方式,然後用$var=null的方式清除。其它時候還是用php5的預設方式吧。另外,php5中對於大陣列的傳遞,建議用 "&" 方式,畢竟節省記憶體空間使用。

當你 unset 一個引用,只是斷開了變數名和變數內容之間的繫結。這並不意味著變數內容被銷燬了。例如:

<?php
$a = 1;
$b =& $a;
unset ($a);
?>

不會 unset $b,只是 $a。

當用 global $var 宣告一個變數時實際上建立了一個到全域性變數的引用。也就是說和這樣做是相同的:

<?php
$var =& $GLOBALS["var"];
?>

這意味著,例如,unset $var 不會 unset 全域性變數。

$this 在一個物件的方法中,$this 永遠是呼叫它的物件的引用。

php中對於地址的指向(類似指標)功能不是由使用者自己來實現的,是由Zend核心實現的,php中引用採用的是"寫時複製"的原理,就是除非發生寫操作,指向同一個地址的變數或者物件是不會被複製的。

通俗的講,如果有下面的程式碼:

$a="ABC";
$b=$a;

其實此時$a與$b都是指向同一記憶體地址,而並不是$a與$b佔用不同的記憶體。

如果在上面的程式碼基礎上再加上如下程式碼:

$a="EFG";

由於$a與$b所指向的記憶體的資料要重新寫一次了,此時Zend核心會自動判斷 自動為$b生產一個$a的資料複製,重新申請一塊記憶體進行儲存。


相關文章