首頁 >php教程 >php手册 >深入分析PHP引用(&)_php基礎

深入分析PHP引用(&)_php基礎

WBOY
WBOY原創
2016-05-16 09:00:022506瀏覽

引用是什麼
在 php 中引用意味著用不同的名字存取同一個變數內容。這並不像 c 的指針,替代的是,引用是符號表別名。注意在 php 中,變數名和變數內容是不一樣的,因此同樣的內容可以有不同的名字。最接近的比喻是 unix 的檔案名稱和檔案本身-變數名稱是目錄條目,而變數內容則是檔案本身。引用可以被看作是 unix 檔案系統中的 hardlink。

引用做什麼
php 的引用允許用兩個變數來指向同一個內容。意思是,當這樣做時:

<?php
$a =& $b;
?>

這表示 $a 和 $b 指向了同一個變數。

note:

$a 和 $b 在這裡是完全相同的,這並不是 $a 指向了 $b 或相反,而是 $a 和 $b 指向了同一個地方。

note:

如果具有引用的陣列被拷貝,其值不會解除引用。對於數組傳值給函數也是如此。

note:

如果對一個未定義的變數進行引用賦值、引用參數傳遞或參考傳回,則會自動建立該變數。

example #1 對未定義的變數使用引用

<?php
function foo(&$var) { }

foo($a); // $a is "created" and assigned to null

$b = array();
foo($b['b']);
var_dump(array_key_exists('b', $b)); // bool(true)

$c = new stdclass;
foo($c->d);
var_dump(property_exists($c, 'd')); // bool(true)
?>

同樣的語法可以用在函數中,它回傳引用,以及用在 new 運算子中(php 4.0.4 以及以後版本):

<?php
$bar =& new fooclass();
$foo =& find_var($bar);
?>


自 php 5 起,new 自動傳回引用,因此在此使用 =& 已經過時了並且會產生 e_strict 層級的訊息。

note:

不用 & 運算子導致物件產生了一個拷貝。如果在類別中使用 $this,它將作用於該類別目前的實例。沒有用 & 的賦值將拷貝這個實例(例如物件)並且 $this 將作用於這個拷貝上,這並不總是想要的結果。由於效能和記憶體消耗的問題,通常只想工作在一個實例上面。

儘管可以用 @ 運算子來抑制建構函式中的任何錯誤訊息,例如用 @new,但用 &new 語句時這不起效果。這是 zend 引擎的一個限制並且會導致一個解析錯誤。

warning

如果在一個函數內部給一個宣告為 global 的變數賦於一個引用,則該引用只在函數內部可見。可以透過使用 $globals 陣列來避免這一點。

example #2 在函數內引用全域變數

<?php
$var1 = "example variable";
$var2 = "";

function global_references($use_globals)
{
 global $var1, $var2;
 if (!$use_globals) {
  $var2 =& $var1; // visible only inside the function
 } else {
  $globals["var2"] =& $var1; // visible also in global context
 }
}

global_references(false);
echo "var2 is set to '$var2'\n"; // var2 is set to ''
global_references(true);
echo "var2 is set to '$var2'\n"; // var2 is set to 'example variable'
?>

把 global $var; 當成是 $var =& $globals['var']; 的簡寫。從而將其它引用賦給 $var 只改變了本地變數的引用。
note:

如果在 foreach 語句中給一個具有引用的變數賦值,被引用的物件也被改變。

example #3 引用與 foreach 語句

<?php
$ref = 0;
$row =& $ref;
foreach (array(1, 2, 3) as $row) {
 // do something
}
echo $ref; // 3 - last element of the iterated array
?>

引用做的第二件事是用引用傳遞變數。這是透過在函數內建立一個本地變數並且該變數在呼叫範圍內引用了同一個內容來實現的。例如:

<?php
function foo(&$var)
{
 $var++;
}

$a=5;
foo($a);
?>

將使 $a 變成 6。這是因為在 foo 函數中變數 $var 指向了和 $a 指向的同一個內容。更多詳細解釋請見引用傳遞。

引用做的第三件事是引用回傳。

引用不是什麼
如前所述,引用不是指標。這意味著下面的結構不會產生預期的效果:

<?php
function foo(&$var)
{
 $var =& $globals["baz"];
}
foo($bar);
?>

這將使 foo 函數中的 $var 變數在函數呼叫時和 $bar 綁定在一起,但接著又被重新綁定到了 $globals["baz"] 上面。不可能透過引用機制將$bar 在函數呼叫範圍內綁定到別的變數上面,因為在函數foo 中並沒有變數$bar(它被表示為$var,但是$var 只有變數內容而沒有呼叫符號表中的名字到值的綁定)。可以使用引用返回來引用被函數選擇的變數。

引用傳遞
可以將一個變數透過引用傳遞給函數,這樣函數就可以修改其參數的值。文法如下:

<?php
function foo(&$var)
{
 $var++;
}

$a=5;
foo($a);
// $a is 6 here
?>

注意函數呼叫時沒有引用符號-只有函數定義中有。光是函數定義就足夠使參數透過引用來正確傳遞了。在最近版本的 php 中如果把 & 用在 foo(&$a); 中會得到一條警告說「call-time pass-by-reference」已經過時了。

以下內容可以透過引用傳遞:

變量,例如 foo($a)
new 語句,例如 foo(new foobar())
從函數傳回的引用,例如:

<?php
function &bar()
{
 $a = 5;
 return $a;
}
foo(bar());
?>

詳細解釋請參閱引用回傳。
任何其它表達式都不能透過引用傳遞,結果未定義。例如下面引用傳遞的例子是無效的:

<?php
function bar() // note the missing &
{
 $a = 5;
 return $a;
}
foo(bar()); // 自 php 5.0.5 起导致致命错误
foo($a = 5) // 表达式,不是变量
foo(5) // 导致致命错误
?>

這些條件是 php 4.0.4 以及以後版本有的。

引用回傳
引用回傳用在當想用函數找到引用應該被綁定在哪一個變數上面。不要用返回引用來增加效能,引擎足夠聰明來自己進行優化。僅在有合理的技術原因時才返回引用!若要返回引用,請使用此語法:

<?php
class foo {
 public $value = 42;

 public function &getvalue() {
  return $this->value;
 }
}

$obj = new foo;
$myvalue = &$obj->getvalue(); // $myvalue is a reference to $obj->value, which is 42.
$obj->value = 2;
echo $myvalue;    // prints the new value of $obj->value, i.e. 2.
?>

本例中 getvalue 函數所傳回的物件的屬性將被賦值,而不是拷貝,就和沒有用引用語法一樣。

note: 和參數傳遞不同,這裡必須在兩個地方都用& 符號——指出返回的是一個引用,而不是通常的一個拷貝,同樣也指出$myvalue 是作為引用的綁定,而不是通常的賦值。

note: 如果試圖這樣從函數傳回引用:return ($this->value);,這將不會起作用,因為在試圖傳回一個表達式的結果而不是一個引用的變數。只能從函數傳回引用變數——沒別的方法。如果程式碼試圖傳回動態表達式或 new 運算子的結果,自 php 4.4.0 和 php 5.1.0 起會發出一條 e_notice 錯誤。

<?php
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。

取消引用
當 unset 一個引用,只是斷開了變數名稱和變數內容之間的綁定。這並不意味著變數內容被銷毀了。例如:

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

不會 unset $b,只是 $a。

再拿這個和 unix 的 unlink 呼叫來類比一下可能有助於理解。

引用定位
許多 php 的語法結構是透過引用機制實現的,所以上述有關引用綁定的一切也都適用於這些結構。一些結構,例如引用傳遞和返回,已經在上面提到了。其它使用引用的結構有:

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

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

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

使用unset($a)與$a=null的結果是不一樣的。如果該區塊記憶體只有$a一個映射,那麼unset($a)與$a=null等價,該記憶體的參考計數變成0,被自動回收;如果該區塊記憶體有$a和$b兩個映射,那麼unset($a)會導致$a=null且$b不變的情況,而$a=null會導致$a=$b=null的情況。

原因:某變數賦值為null,將導致該變數對應的記憶體區塊的參考計數直接置為0,並自動回收。

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

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

下面再來個小插曲php中對於位址的指向(類似指標)功能不是由使用者自己來實現的,是由zend核心實現的,php中引用採用的是「寫時拷貝」的原理,就是除非發生寫入操作,指向同一個位址的變數或物件是不會被拷貝的。

通俗的講

1:如果有下面的程式碼

<?ph
$a="ABC";
$b=$a;
?>
 

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

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

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

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn