首頁  >  文章  >  後端開發  >  PHP引用的意思及用法

PHP引用的意思及用法

墨辰丷
墨辰丷原創
2018-06-13 10:22:581374瀏覽

本文這裡向大家全面展示了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[&#39;b&#39;]);
var_dump(array_key_exists(&#39;b&#39;, $b)); // bool(true)
$c = new StdClass;
foo($c->d);
var_dump(property_exists($c, &#39;d&#39;)); // 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 &#39;$var2&#39;\n"; // var2 is set to &#39;&#39;
global_references(true);
echo "var2 is set to &#39;$var2&#39;\n"; // var2 is set to &#39;Example variable&#39;
?>

把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的数据拷贝,重新申请一块内存进行存储。

总结:以上就是本篇文的全部内容,希望能对大家的学习有所帮助。

相关推荐:

php实现去除字符串中空字符的方法

PHP实现批量生成各种尺寸Logo的方法

PHP中array_fill函数定义多维数组的方法

以上是PHP引用的意思及用法的詳細內容。更多資訊請關注PHP中文網其他相關文章!

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