PHP的引用详解


Posted in PHP onFebruary 22, 2015

引用是什么

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

以上就是关于PHP引用的全部内容了,希望大家能够喜欢。

PHP 相关文章推荐
php max_execution_time执行时间问题
Jul 17 PHP
解析PHP获取当前网址及域名的实现代码
Jun 23 PHP
使用PHP下载CSS文件中的图片的代码
Sep 24 PHP
smarty模板中使用get、post、request、cookies、session变量的方法
Apr 24 PHP
PHP实现自动对图片进行滚动显示的方法
Mar 12 PHP
支付宝接口开发集成支付环境小结
Mar 17 PHP
Linux系统下PHP-FPM的安装和配置教程
Aug 17 PHP
PHP中file_get_contents函数抓取https地址出错的解决方法(两种方法)
Sep 22 PHP
php批量删除超链接的实现方法
Oct 19 PHP
PHP数组遍历的几种常见方式总结
Feb 15 PHP
php装饰者模式简单应用案例分析
Oct 23 PHP
php实现统计IP数及在线人数的示例代码
Jul 22 PHP
PHP实现加密的几种方式介绍
Feb 22 #PHP
php使用parse_url和parse_str解析URL
Feb 22 #PHP
php的闭包(Closure)匿名函数详解
Feb 22 #PHP
PHP 实现代码复用的一个方法 traits新特性
Feb 22 #PHP
在Windows XP下安装Apache+MySQL+PHP环境
Feb 22 #PHP
PHP+APACHE实现网址伪静态
Feb 22 #PHP
php数组键名技巧小结
Feb 17 #PHP
You might like
Linux平台php命令行程序处理管道数据的方法
2016/11/10 PHP
php依赖注入知识点详解
2019/09/23 PHP
javaScript如何生成xmlhttp
2013/12/16 Javascript
JS的get和set使用示例
2014/02/20 Javascript
javascript 事件处理示例分享
2014/12/31 Javascript
详解javascript中的事件处理
2015/11/06 Javascript
深入浅析javascript中的作用域(推荐)
2016/07/19 Javascript
AngularJS使用ng-class动态增减class样式的方法示例
2017/05/18 Javascript
微信小程序实现的贪吃蛇游戏【附源码下载】
2018/01/03 Javascript
vue车牌号校验和银行校验实战
2019/01/23 Javascript
原生JS实现多条件筛选
2020/08/19 Javascript
JavaScript TAB栏切换效果的示例
2020/11/05 Javascript
[01:04]DOTA2上海特锦赛现场采访 FreeAgain遭众解说围攻
2016/03/25 DOTA
[02:38]2018年度DOTA2最佳劣单位选手-完美盛典
2018/12/17 DOTA
[57:09]DOTA2-DPC中国联赛 正赛 Phoenix vs Dynasty BO3 第一场 1月26日
2021/03/11 DOTA
python实现将读入的多维list转为一维list的方法
2018/06/28 Python
使用Python监视指定目录下文件变更的方法
2018/10/15 Python
对python自动生成接口测试的示例讲解
2018/11/30 Python
使用 Python 玩转 GitHub 的贡献板(推荐)
2019/04/04 Python
Python Flask框架模板操作实例分析
2019/05/03 Python
HTML5的表单(绝对特别强大的功能)使用示例
2013/06/20 HTML / CSS
html5写一个BUI折叠菜单插件的实现方法
2019/09/11 HTML / CSS
Darphin迪梵官网: 来自巴黎,植物和精油调制的护肤品牌
2016/10/11 全球购物
德国低价购买灯具和家具网站:Style-home.de
2016/11/25 全球购物
MANGO官方网站:西班牙芒果服装品牌
2017/01/15 全球购物
丹麦优惠购物网站:PLUSSHOP
2019/03/24 全球购物
毕业生应聘幼儿园的自荐信
2013/11/20 职场文书
装修致歉信
2014/01/15 职场文书
人力管理专业毕业生求职信
2014/02/27 职场文书
大学生个人自荐信样本
2014/03/02 职场文书
法人任命书范本
2014/06/04 职场文书
讲文明知礼仪演讲稿
2014/09/13 职场文书
党员思想汇报材料
2014/12/19 职场文书
三年级学生评语大全
2014/12/26 职场文书
学校党风廉政建设调研报告
2015/01/01 职场文书
英文感谢信范文
2015/01/21 职场文书