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

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

May 16, 2016 am 09:00 AM
php 引用

引用是什麼
在 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

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳圖形設置
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您聽不到任何人,如何修復音頻
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解鎖Myrise中的所有內容
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

適用於 Ubuntu 和 Debian 的 PHP 8.4 安裝和升級指南 適用於 Ubuntu 和 Debian 的 PHP 8.4 安裝和升級指南 Dec 24, 2024 pm 04:42 PM

PHP 8.4 帶來了多項新功能、安全性改進和效能改進,同時棄用和刪除了大量功能。 本指南介紹如何在 Ubuntu、Debian 或其衍生版本上安裝 PHP 8.4 或升級到 PHP 8.4

CakePHP 日期和時間 CakePHP 日期和時間 Sep 10, 2024 pm 05:27 PM

為了在 cakephp4 中處理日期和時間,我們將使用可用的 FrozenTime 類別。

CakePHP 檔案上傳 CakePHP 檔案上傳 Sep 10, 2024 pm 05:27 PM

為了進行文件上傳,我們將使用表單助理。這是文件上傳的範例。

討論 CakePHP 討論 CakePHP Sep 10, 2024 pm 05:28 PM

CakePHP 是 PHP 的開源框架。它旨在使應用程式的開發、部署和維護變得更加容易。 CakePHP 基於類似 MVC 的架構,功能強大且易於掌握。模型、視圖和控制器 gu

CakePHP 建立驗證器 CakePHP 建立驗證器 Sep 10, 2024 pm 05:26 PM

可以透過在控制器中新增以下兩行來建立驗證器。

CakePHP 日誌記錄 CakePHP 日誌記錄 Sep 10, 2024 pm 05:26 PM

登入 CakePHP 是一項非常簡單的任務。您只需使用一項功能即可。您可以記錄任何後台程序(如 cronjob)的錯誤、異常、使用者活動、使用者採取的操作。在 CakePHP 中記錄資料很容易。提供了 log() 函數

如何設定 Visual Studio Code (VS Code) 進行 PHP 開發 如何設定 Visual Studio Code (VS Code) 進行 PHP 開發 Dec 20, 2024 am 11:31 AM

Visual Studio Code,也稱為 VS Code,是一個免費的原始碼編輯器 - 或整合開發環境 (IDE) - 可用於所有主要作業系統。 VS Code 擁有大量針對多種程式語言的擴展,可以輕鬆編寫

CakePHP 快速指南 CakePHP 快速指南 Sep 10, 2024 pm 05:27 PM

CakePHP 是一個開源MVC 框架。它使應用程式的開發、部署和維護變得更加容易。 CakePHP 有許多函式庫可以減少大多數常見任務的過載。

See all articles