服务报价 | 域名主机 | 网络营销 | 软件工具| [加入收藏]
 热线电话: #
当前位置: 主页 > php教程 > php教程 >

PHP引用传递与引用

时间:2016-02-23 11:49来源: 作者: 点击:
在 PHP 中引用意味着用不同的名字访问同一个变量内容,这并不像 C 的指针,替代的是,引用是符号表别名,注意在 PHP 中,变量名和变量内容是不一样的,因此同样的内容可以有不同的名字,最
在 PHP 中引用意味着用不同的名字访问同一个变量内容,这并不像 C 的指针,替代的是,引用是符号表别名,注意在 PHP 中,变量名和变量内容是不一样的,因此同样的内容可以有不同的名字,最接近的比喻是 Unix 的文件名和文件本身——变量名是目录条目,而变量内容则是文件本身,引用可以被看作是 Unix 文件系统中的 hardlink。
 
引用做什么
 
PHP 的引用允许用两个变量来指向同一个内容。
 
当 $a =& $b; 时 $a 和 $b 指向了同一个变量。
 
提示:$a 和 $b 在这里是完全相同的,这并不是 $a 指向了 $b 或者相反,而是 $a 和 $b 指向了同一个地方。
 
可以将一个变量通过引用传递给函数,这样该函数就可以修改其参数的值,语法如下:
 
<?php  
function foo(&$var)  
{  
    $var++;  
}  
   
$a=5;  
foo($a);  
echo $a; 
// 输出的是:6 
PHP引用符&
 
关于php的引用(就是在变量或者函数、对象等前面加上&符号)的作用,我们先看下面这个程序。代码如下:
 
<?php 
$a = 100; //声明变量a 
$b = &$a; //声明变量b,引用自变量a 
echo "$a <br />";   
echo "$b <br />"; 
$a++; //变量a自增1 
echo "$a <br />"; 
echo "$b <br />";//查看变量b,也增加了1,说明使用的是同一存储单元 
?> 
//程序运行结果: 
//100  
//100  
//101  
//101 
很多人误解php中的引用跟C当中的指针一样,事实上并非如此,而且很大差别。C语言中的指针除了在数组传递过程中不用显式申明外,其他都需要使用*进行定义,而php中对于地址的指向(类似指针)功能不是由用户自己来实现的,是由Zend核心实现的,php中引用采用的是“写时拷贝”的原理,就是除非发生写操作,指向同一个地址的变量或者对象是不会被拷贝的。
 
php默认为传值传递,代码如下:
 
<?php     
$a = 20; 
$b = $a; 
$a = $a + 10;  
echo $a.' and '.$b;  
?> 
//程序运行结果: 
//30 and 20 
要是想变为地址传递需要加&,既代码如下:
 
<?php 
$a = 20; 
$b = &$a;  
$a = $a + 10;  
echo $a.' and '.$b;  
?> 
就是说,&把$a的地址传给了$b,这样的话这两个变量现在共享一个内存的存储区域,就是说它们的值是一样的。
 
同样的语法可以用在函数中,它返回引用,以及用在 new 运算符中,代码如下:
 
<?php  
$bar =& new fooclass();  
$foo =& find_var($bar);  
?> 
引用做的第二件事是用引用传递变量,这是通过在函数内建立一个本地变量,并且该变量在呼叫范围内引用了同一个内容来实现的,说的通俗点就是一个函数的参数是一个本地变量的引用,下面再举例说明一下:
 
<?php 
function foo(&$val1, $val2) { 
 $val1 += 1; 
 $val2 += 1; 
$a=5; 
$b=10; 
foo($a,$b); 
echo $a; 
echo $b; 
?> 
运行这段代码是给函数传递两个参数,一个是引用$a的内容,一个是$b的值,在执行此函数后,发现$a的内容改变了,而$b的内容则没有变化。
 
PHP引用以及误区
 
PHP中的引用可以理解成变量的别名,由于PHP的变量名是存储在符号表(symbol table)中的,变量内容是存储在堆中,引用就是用符号表中的不同符号(symbol)名称来访问同一存储内容,和Unix文件系统中的hardlink是同一个概念,比如实例代码如下:
 
<?php 
$a = 1; 
$b = &$a; //$a与$b指向同一内容 
$b = 2; 
echo $b; //2 
echo $a; //2 
?> 
传递引用
 
引用传递很简单,就是一个「&」符号,比如如下代码:
 
<?php 
function foo(&$a) { 
  $a = 2; 
$b = 1; 
foo($b); 
echo $b; //2 
?> 
返回引用
 
大多数情况下并不需要返回引用来提高性能,zend引擎会自己进行优化,但是如果你非得返回引用得话,可以按照以下方式来返回引用,代码如下:
 
<?php 
class foo { 
    public $value = 42; 
    public function &getValue() { // 需要一个"&" 
        return $this->value; 
    } 
$obj = new foo; 
$myValue = &$obj->getValue(); // 还需要一个"&",$myValue是对class foo中的$value的引用 
$obj->value = 2;              // 修改对象的$value属性 
echo $myValue;                // 输出2,$myValue与class foo中的$value值相同 
?> 
与指针的区别
 
引用与指针很像,但是其并不是指针,看如下的代码:
 
<?php 
    $a = 0; 
    $b = &a; 
    echo $a; //0 
    unset($b); 
    echo $a; //0 
?> 
由于$b只是$a的别名,所以即使$b被释放了,$a没有任何影响,但是指针可不是这样的,看如下代码:
 
#include <stdio.h> 
int main(int argc, char const *argv[]) { 
    int a = 0; 
    int* b = &a; 
    printf("%in", a); //0 
    free(b); 
    printf("%in", a); //*** error for object 0x7fff6350da08: pointer being freed was not allocated 
由于b是指向a的指针,所以释放了b的内存之后,再访问a就会出现错误,比较明显的说明了PHP引用与C指针的区别。
 
对象与引用
 
在PHP中使用对象的时候,大家总是被告知“对象是按照引用传递的”,其实这是个误区,PHP的对象变量存储的是此对象的一个标示符,在传递对象的时候,其实传递的就是这个标示符,而并不是引用,看如下代码:
 
<?php 
$a = new A; 
$b = $a;     
$b->testA = 2; 
/* 
 * 此时$a,$b的关系: 
 *        +-----------+      +-----------------+ 
 * $a --> | object id | ---> | object(Class A) | 
 *        +-----------+      +-----------------+ 
 *                               ^ 
 *        +-----------+          | 
 * $b --> | object id | ---------+ 
 *        +-----------+     
 * 
 * 
 */ 
$c = new B; 
$a = $c; 
$a->testB = "Changed Class B"; 
/* 
 * 此时$a,$b,$c的关系: 
 *        +-----------+      +-----------------+ 
 * $b --> | object id | ---> | object(Class A) | 
 *        +-----------+      +-----------------+ 
 *                                
 *        +------------+           
 * $a --> | object id2 | -------------+ 
 *        +------------+              | 
 *                                    v 
 *        +------------+      +-----------------+ 
 * $c --> | object id2 | ---> | object(Class B) | 
 *        +------------+      +-----------------+ 
 */ 
echo "object a: "; var_dump($a); //["testB"]=> string(15) "Changed Class B" 
echo "object b: "; var_dump($b); //["testA"] => int(2) 
echo "object c: "; var_dump($c); //["testB"]=> string(15) "Changed Class B" 
?> 
如果对象是按照引用传递的,那么$a, $b, $c输出的内容应该一样,事实上结果并非如此,看下面通过引用传递对象的列子:
 
<?php 
$aa = new A; 
$bb = &$aa;  // 引用  
$bb->testA = 2; 
/* 
 * 此时$aa, $bb的关系: 
 * 
 *         +-----------+      +-----------------+ 
 * $bb --> | object id | ---> | object(Class A) | 
 *         +-----------+      +-----------------+ 
 *              ^                   
 *              | 
 * $aa ---------+  
 * 
 * 
 */ 
$cc = new B; 
$aa = $cc; 
$aa->testB = "Changed Class B"; 
/* 
 * 此时$aa, $bb, $cc的关系: 
 * 
 *         +-----------+      +-----------------+ 
 *         | object id | ---> | object(Class A) | 
 *         +-----------+      +-----------------+ 
 *               
 * $bb ---->-----+       
 *               | 
 * $aa ---->-----+ 
 *               |   
 *               v    
 *         +------------+       
 *         | object id2 | --------------+  
 *         +------------+               | 
 *                                      v 
 *         +------------+      +-----------------+ 
 * $cc --> | object id2 | ---> | object(Class B) | 
 *         +------------+      +-----------------+ 
 */ 
echo "object aa: "; var_dump($aa); //["testB"]=>string(15) "Changed Class B" 
echo "object bb: "; var_dump($bb); //["testB"]=>string(15) "Changed Class B" 
echo "object cc: "; var_dump($cc); //["testB"]=>string(15) "Changed Class B" 
?> 
此时$aa,$bb,$cc三者内容完全一样,所以可以看出对象并不是按照引用传递,要尽快走出这个误区.
(责任编辑:最模板)
顶一下
(0)
0%
踩一下
(0)
0%
------分隔线----------------------------
栏目列表
热点内容