PHP的引用詳解
引用是什么
在 PHP 中引用意味著用不同的名字訪問同一個(gè)變量內(nèi)容。這并不像 C 的指針,替代的是,引用是符號(hào)表別名。注意在 PHP 中,變量名和變量內(nèi)容是不一樣的,因此同樣的內(nèi)容可以有不同的名字。最接近的比喻是 Unix 的文件名和文件本身——變量名是目錄條目,而變量內(nèi)容則是文件本身。引用可以被看作是 Unix 文件系統(tǒng)中的 hardlink。
引用做什么
PHP 的引用允許用兩個(gè)變量來指向同一個(gè)內(nèi)容。意思是,當(dāng)這樣做時(shí):
<?php
$a =& $b;
?>
這意味著 $a 和 $b 指向了同一個(gè)變量。
Note:
$a 和 $b 在這里是完全相同的,這并不是 $a 指向了 $b 或者相反,而是 $a 和 $b 指向了同一個(gè)地方。
Note:
如果具有引用的數(shù)組被拷貝,其值不會(huì)解除引用。對(duì)于數(shù)組傳值給函數(shù)也是如此。
Note:
如果對(duì)一個(gè)未定義的變量進(jìn)行引用賦值、引用參數(shù)傳遞或引用返回,則會(huì)自動(dòng)創(chuàng)建該變量。
Example #1 對(duì)未定義的變量使用引用
<?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)
?>
同樣的語法可以用在函數(shù)中,它返回引用,以及用在 new 運(yùn)算符中(PHP 4.0.4 以及以后版本):
<?php
$bar =& new fooclass();
$foo =& find_var($bar);
?>
自 PHP 5 起,new 自動(dòng)返回引用,因此在此使用 =& 已經(jīng)過時(shí)了并且會(huì)產(chǎn)生 E_STRICT 級(jí)別的消息。
Note:
不用 & 運(yùn)算符導(dǎo)致對(duì)象生成了一個(gè)拷貝。如果在類中用 $this,它將作用于該類當(dāng)前的實(shí)例。沒有用 & 的賦值將拷貝這個(gè)實(shí)例(例如對(duì)象)并且 $this 將作用于這個(gè)拷貝上,這并不總是想要的結(jié)果。由于性能和內(nèi)存消耗的問題,通常只想工作在一個(gè)實(shí)例上面。
盡管可以用 @ 運(yùn)算符來抑制構(gòu)造函數(shù)中的任何錯(cuò)誤信息,例如用 @new,但用 &new 語句時(shí)這不起效果。這是 Zend 引擎的一個(gè)限制并且會(huì)導(dǎo)致一個(gè)解析錯(cuò)誤。
Warning
如果在一個(gè)函數(shù)內(nèi)部給一個(gè)聲明為 global 的變量賦于一個(gè)引用,該引用只在函數(shù)內(nèi)部可見??梢酝ㄟ^使用 $GLOBALS 數(shù)組避免這一點(diǎn)。
Example #2 在函數(shù)內(nèi)引用全局變量
<?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; 當(dāng)成是 $var =& $GLOBALS['var']; 的簡寫。從而將其它引用賦給 $var 只改變了本地變量的引用。
Note:
如果在 foreach 語句中給一個(gè)具有引用的變量賦值,被引用的對(duì)象也被改變。
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
?>
引用做的第二件事是用引用傳遞變量。這是通過在函數(shù)內(nèi)建立一個(gè)本地變量并且該變量在呼叫范圍內(nèi)引用了同一個(gè)內(nèi)容來實(shí)現(xiàn)的。例如:
<?php
function foo(&$var)
{
$var++;
}
$a=5;
foo($a);
?>
將使 $a 變成 6。這是因?yàn)樵?foo 函數(shù)中變量 $var 指向了和 $a 指向的同一個(gè)內(nèi)容。更多詳細(xì)解釋見引用傳遞。
引用做的第三件事是引用返回。
引用不是什么
如前所述,引用不是指針。這意味著下面的結(jié)構(gòu)不會(huì)產(chǎn)生預(yù)期的效果:
<?php
function foo(&$var)
{
$var =& $GLOBALS["baz"];
}
foo($bar);
?>
這將使 foo 函數(shù)中的 $var 變量在函數(shù)調(diào)用時(shí)和 $bar 綁定在一起,但接著又被重新綁定到了 $GLOBALS["baz"] 上面。不可能通過引用機(jī)制將 $bar 在函數(shù)調(diào)用范圍內(nèi)綁定到別的變量上面,因?yàn)樵诤瘮?shù) foo 中并沒有變量$bar(它被表示為 $var,但是 $var 只有變量內(nèi)容而沒有調(diào)用符號(hào)表中的名字到值的綁定)??梢允褂靡梅祷貋硪帽缓瘮?shù)選擇的變量。
引用傳遞
可以將一個(gè)變量通過引用傳遞給函數(shù),這樣該函數(shù)就可以修改其參數(shù)的值。語法如下:
<?php
function foo(&$var)
{
$var++;
}
$a=5;
foo($a);
// $a is 6 here
?>
注意在函數(shù)調(diào)用時(shí)沒有引用符號(hào)——只有函數(shù)定義中有。光是函數(shù)定義就足夠使參數(shù)通過引用來正確傳遞了。在最近版本的 PHP 中如果把 & 用在 foo(&$a); 中會(huì)得到一條警告說“Call-time pass-by-reference”已經(jīng)過時(shí)了。
以下內(nèi)容可以通過引用傳遞:
變量,例如 foo($a)
New 語句,例如 foo(new foobar())
從函數(shù)中返回的引用,例如:
<?php
function &bar()
{
$a = 5;
return $a;
}
foo(bar());
?>
詳細(xì)解釋見引用返回。
任何其它表達(dá)式都不能通過引用傳遞,結(jié)果未定義。例如下面引用傳遞的例子是無效的:
<?php
function bar() // Note the missing &
{
$a = 5;
return $a;
}
foo(bar()); // 自 PHP 5.0.5 起導(dǎo)致致命錯(cuò)誤
foo($a = 5) // 表達(dá)式,不是變量
foo(5) // 導(dǎo)致致命錯(cuò)誤
?>
這些條件是 PHP 4.0.4 以及以后版本有的。
引用返回
引用返回用在當(dāng)想用函數(shù)找到引用應(yīng)該被綁定在哪一個(gè)變量上面時(shí)。不要用返回引用來增加性能,引擎足夠聰明來自己進(jìn)行優(yōu)化。僅在有合理的技術(shù)原因時(shí)才返回引用!要返回引用,使用此語法:
<?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 函數(shù)所返回的對(duì)象的屬性將被賦值,而不是拷貝,就和沒有用引用語法一樣。
Note: 和參數(shù)傳遞不同,這里必須在兩個(gè)地方都用 & 符號(hào)——指出返回的是一個(gè)引用,而不是通常的一個(gè)拷貝,同樣也指出 $myValue 是作為引用的綁定,而不是通常的賦值。
Note: 如果試圖這樣從函數(shù)返回引用:return ($this->value);,這將不會(huì)起作用,因?yàn)樵谠噲D返回一個(gè)表達(dá)式的結(jié)果而不是一個(gè)引用的變量。只能從函數(shù)返回引用變量——沒別的方法。如果代碼試圖返回一個(gè)動(dòng)態(tài)表達(dá)式或 new 運(yùn)算符的結(jié)果,自 PHP 4.4.0 和 PHP 5.1.0 起會(huì)發(fā)出一條 E_NOTICE 錯(cuò)誤。
<?php
function &test(){
static $b=0;//申明一個(gè)靜態(tài)變量
$b=$b+1;
echo $b;
return $b;
}
$a=test();//這條語句會(huì)輸出$b的值為1
$a=5; $a=test();//這條語句會(huì)輸出$b的值為2
$a=&test();//這條語句會(huì)輸出$b的值為3
$a=5; $a=test();//這條語句會(huì)輸出$b的值為6
?>
$a=test()方式調(diào)用函數(shù),只是將函數(shù)的值賦給$a而已,而$a做任何改變化,都不會(huì)影響到函數(shù)中的$b,而通過$a=&test()方式調(diào)用函數(shù)呢, 他的作用是將return $b中的$b變量的內(nèi)存地址與$a變量的內(nèi)存地址指向了同一個(gè)地方,即產(chǎn)生了相當(dāng)于這樣的效果($a=&b;) 所以改變$a的值,也同時(shí)改變了$b的值,所以在執(zhí)行了 $a=&test(); $a=5; 以后,$b的值變?yōu)榱?。
取消引用
當(dāng) unset 一個(gè)引用,只是斷開了變量名和變量內(nèi)容之間的綁定。這并不意味著變量內(nèi)容被銷毀了。例如:
<?php
$a = 1;
$b =& $a;
unset($a);
?>
不會(huì) unset $b,只是 $a。
再拿這個(gè)和 Unix 的 unlink 調(diào)用來類比一下可能有助于理解。
引用定位
許多 PHP 的語法結(jié)構(gòu)是通過引用機(jī)制實(shí)現(xiàn)的,所以上述有關(guān)引用綁定的一切也都適用于這些結(jié)構(gòu)。一些結(jié)構(gòu),例如引用傳遞和返回,已經(jīng)在上面提到了。其它使用引用的結(jié)構(gòu)有:
global 引用
當(dāng)用 global $var 聲明一個(gè)變量時(shí)實(shí)際上建立了一個(gè)到全局變量的引用。也就是說和這樣做是相同的:
<?php
$var =& $GLOBALS["var"];
?>
這意味著,例如,unset $var 不會(huì) unset 全局變量。
使用unset($a)與$a=null的結(jié)果是不一樣的。如果該塊內(nèi)存只有$a一個(gè)映射,那么unset($a)與$a=null等價(jià),該內(nèi)存的引用計(jì)數(shù)變?yōu)?,被自動(dòng)回收;如果該塊內(nèi)存有$a和$b兩個(gè)映射,那么unset($a)將導(dǎo)致$a=null且$b不變的情況,而$a=null會(huì)導(dǎo)致$a=$b=null的情況。
原因:某變量賦值為null,將導(dǎo)致該變量對(duì)應(yīng)的內(nèi)存塊的引用計(jì)數(shù)直接置為0,被自動(dòng)回收。
$this
在一個(gè)對(duì)象的方法中,$this 永遠(yuǎn)是調(diào)用它的對(duì)象的引用。
引用的作用
如果程序比較大,引用同一個(gè)對(duì)象的變量比較多,并且希望用完該對(duì)象后手工清除它,個(gè)人建議用 "&" 方式,然后用$var=null的方式清除. 其它時(shí)候還是用php5的默認(rèn)方式吧. 另外, php5中對(duì)于大數(shù)組的傳遞,建議用 "&" 方式, 畢竟節(jié)省內(nèi)存空間使用。
下面再來個(gè)小插曲 php中對(duì)于地址的指向(類似指針)功能不是由用戶自己來實(shí)現(xiàn)的,是由Zend核心實(shí)現(xiàn)的,php中引用采用的是“寫時(shí)拷貝”的原理,就是除非發(fā)生寫操作,指向同一個(gè)地址的變量或者對(duì)象是不會(huì)被拷貝的。
通俗的講
1:如果有下面的代碼
<?ph
$a="ABC";
$b=$a;
?>
其實(shí)此時(shí),$a與$b都是指向同一內(nèi)存地址,而并不是$a與$b占用不同的內(nèi)存。
2:如果在上面的代碼基礎(chǔ)上再加上如下代碼
$a="EFG";
由于$a與$b所指向的內(nèi)存的數(shù)據(jù)要重新寫一次了,此時(shí)Zend核心會(huì)自動(dòng)判斷 自動(dòng)為$b生產(chǎn)一個(gè)$a的數(shù)據(jù)拷貝,重新申請(qǐng)一塊內(nèi)存進(jìn)行存儲(chǔ)。
以上就是關(guān)于PHP引用的全部內(nèi)容了,希望大家能夠喜歡。
相關(guān)文章
linux下使用crontab實(shí)現(xiàn)定時(shí)PHP計(jì)劃任務(wù)失敗的原因分析
這篇文章主要介紹了linux下使用crontab實(shí)現(xiàn)定時(shí)PHP計(jì)劃任務(wù)失敗的原因分析,需要的朋友可以參考下2014-07-07
PHP調(diào)用MySQL的存儲(chǔ)過程的實(shí)現(xiàn)代碼
MySQL好像從5.0開始才引入存儲(chǔ)過程,反正以前做應(yīng)用的時(shí)候從沒碰過,不過現(xiàn)在因?yàn)橹饕鲀?nèi)部系統(tǒng)2008-08-08
jQuery+PHP+ajax實(shí)現(xiàn)微博加載更多內(nèi)容列表功能
這篇文章主要介紹了jQuery+PHP+ajax實(shí)現(xiàn)微博加載更多內(nèi)容列表功能,對(duì)于微博開發(fā)來說非常實(shí)用,需要的朋友可以參考下2014-06-06
PHP數(shù)組相加操作及與array_merge的區(qū)別淺析
這篇文章主要給大家介紹了關(guān)于PHP數(shù)組相加操作以及與array_merge的區(qū)別,文中通過示例介紹的很詳細(xì),感興趣的朋友們可以參考學(xué)習(xí),有需要的下面跟著小編一起來學(xué)習(xí)學(xué)習(xí)吧。2016-11-11
PHP正則表達(dá)式處理函數(shù)(PCRE 函數(shù))實(shí)例小結(jié)
這篇文章主要介紹了PHP正則表達(dá)式處理函數(shù)(PCRE 函數(shù)),結(jié)合實(shí)例形式總結(jié)分析了php正則表達(dá)式preg_replace、preg_match、preg_match_all、preg_split及preg_quote等函數(shù)相關(guān)使用技巧,需要的朋友可以參考下2019-05-05

