搜索
首页后端开发php教程php引用(&)详解及注意事项_PHP教程
php引用(&)详解及注意事项_PHP教程Jul 13, 2016 pm 05:49 PM
ampphp函数变量对象引用注意事项符号详解

php的引用(就是在变量或者函数、对象等前面加上&符号)

在PHP 中引用的意思是:不同的名字访问同一个变量内容。
与C语言中的指针是有差别的.C语言中的指针里面存储的是变量的内容,在内存中存放的地址。

1.变量的引用

PHP 的引用允许你用两个变量来指向同一个内容


     $a = " ABC " ;
     $b   =& $a ;
     echo   $a ; // 这里输出:ABC
     echo   $b ; // 这里输出:ABC
     $b = " EFG " ;
     echo   $a ; // 这里$a的值变为EFG 所以输出EFG
     echo   $b ; // 这里输出EFG
?>

 


2.函数的引用传递(传址调用)


 

传址调用我就不多说了 下面直接给出代码

php
     function  test( & $a )
    {
         $a = $a + 100 ;
    }
     $b = 1 ;
     echo   $b ; // 输出1
    test( $b );    // 这里$b传递给函数的其实是$b的变量内容所处的内存地址,通过在函数里改变$a的值 就可以改变$b的值了
     echo   "
" ;
     echo   $b ; // 输出101
?>
 

要注意的是,在这里test(1);的话就会出错,原因自己去想。

 

注意:

    上面的“ test($b); ” 中的$b前面不要加 & 符号,但是在函数“call_user_func_array”中,若要引用传参,就得需要 & 符号,如下代码所示:

 

php

function  a( & $b ){
     $b ++ ;
}
$c = 0 ;

call_user_func_array ( ' a ' , array ( & $c ));

echo   $c ;

// 输出 1

?>
 


3.函数的引用返回


 

先看代码

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();得到的其实不是函数的引用返回,这跟普通的函数调用没有区别 至于原因: 这是PHP的规定
PHP规定通过$a=&test(); 方式得到的才是函数的引用返回
至于什么是引用返回呢(PHP手册上说:引用返回用在当想用函数找到引用应该被绑定在哪一个变量上面时。) 这句狗屁话 害我半天没看懂

用上面的例子来解释就是
$a=test()方式调用函数,只是将函数的值赋给$a而已, 而$a做任何改变 都不会影响到函数中的$b
而通过$a=&test()方式调用函数呢, 他的作用是 将return $b中的 $b变量的内存地址与$a变量的内存地址 指向了同一个地方
即产生了相当于这样的效果($a=&$b;) 所以改变$a的值 也同时改变了$b的值 所以在执行了
$a=&test();
$a=5;
以后,$b的值变为了5

这里是为了让大家理解函数的引用返回才使用静态变量的,其实函数的引用返回多用在对象中


另附一个php官方例子:

This is the way how we  use  pointer to access variable inside the  class .

php
class  talker{

     private   $data   =   ' Hi ' ;

     public   function   &  get(){
         return   $this -> data;
    }
  
     public   function  out(){
         echo   $this -> data;
    }  

}

$aa   =   new  talker();
$d   =   & $aa -> get();

$aa -> out();
$d   =   ' How ' ;
$aa -> out();
$d   =   ' Are ' ;
$aa -> out();
$d   =   ' You ' ;
$aa -> out();
?>

the output is  " HiHowAreYou "

 

 

4.对象的引用


 

php
     class  a{
         var   $abc = " ABC " ;
    }
     $b = new  a;
     $c = $b ;
     echo   $b -> abc; // 这里输出ABC
     echo   $c -> abc; // 这里输出ABC
     $b -> abc = " DEF " ;
     echo   $c -> abc; // 这里输出DEF
?>

以上代码是在PHP5中的运行效果

 

在PHP5中 对象的赋值是个引用的过程。上列中$b=new a; $c=$b; 其实等效于$b=new a; $c=&$b;
PHP5中默认就是通过引用来调用对象, 但有时你可能想建立一个对象的副本,并希望原来的对象的改变不影响到副本 . 为了这样的目的,PHP5定义了一个特殊的方法,称为__clone。

自 PHP 5 起,new 自动返回引用,因此在此使用 =& 已经过时了并且会产生 E_STRICT 级别的消息。
 

 

在php4中,对象的赋值是个拷贝过程,

如:$b=new a,其中new a产生的是一个匿名的a对象实例,而此时的$b是对这个匿名对象的拷贝。同理$c=$b,也是对$b内容的一个拷贝。所以在php4中,为了节省内存空间,$b=new a 一般会改成引用的模式,即 $b=& new a。

 

下面再来个 官方 提供的例子:

 在php5中,你不需要额外添加什么东西就可到达“对象引用”的功能:

 

php
class  foo{
         protected   $name ;
         function  __construct( $str ){
                 $this -> name  =   $str ;
        }
         function  __toString(){
                 return    ' my name is " ' .   $this -> name  . ' " and I live in " '   .   __CLASS__   .   ' ". '   .   " \n " ;
        }
         function  setName( $str ){
                 $this -> name  =   $str ;
        }
}

class  MasterOne{
         protected   $foo ;
         function  __construct( $f ){
                 $this -> foo  =   $f ;
        }
         function  __toString(){
                 return   ' Master:  '   .   __CLASS__   .   '  | foo:  '   .   $this -> foo  .   " \n " ;
        }
         function  setFooName( $str ){
                 $this -> foo -> setName(  $str  );
        }
}

class  MasterTwo{
         protected   $foo ;
         function  __construct( $f ){
                 $this -> foo  =   $f ;
        }
         function  __toString(){
                 return   ' Master:  '   .   __CLASS__   .   '  | foo:  '   .   $this -> foo  .   " \n " ;
        }
         function  setFooName( $str ){
                 $this -> foo -> setName(  $str  );
        }
}

$bar   =   new  foo( ' bar ' );

print ( " \n " );
print ( " Only Created \$bar and printing \$bar\n " );
print (  $bar  );

print ( " \n " );
print ( " Now \$baz is referenced to \$bar and printing \$bar and \$baz\n " );
$baz   =&   $bar ;
print (  $bar  );

print ( " \n " );
print ( " Now Creating MasterOne and Two and passing \$bar to both constructors\n " );
$m1   =   new  MasterOne(  $bar  );
$m2   =   new  MasterTwo(  $bar  );
print (  $m1  );
print (  $m2  );

print ( " \n " );
print ( " Now changing value of \$bar and printing \$bar and \$baz\n " );
$bar -> setName( ' baz ' );
print (  $bar  );
print (  $baz  );

print ( " \n " );
print ( " Now printing again MasterOne and Two\n " );
print (  $m1  );
print (  $m2  );

print ( " \n " );
print ( " Now changing MasterTwo's foo name and printing again MasterOne and Two\n " );
$m2 -> setFooName(  ' MasterTwo\'s Foo '  );
print (  $m1  );
print (  $m2  );

print ( " Also printing \$bar and \$baz\n " );
print (  $bar  );
print (  $baz  );
?>

 

输出:

Only Created  $bar  and printing  $bar
my name is  " bar "  and I live in  " foo " .

Now  $baz  is referenced to  $bar  and printing  $bar  and  $baz
my name is  " bar "  and I live in  " foo " .

Now Creating MasterOne and Two and passing  $bar  to both constructors
Master :  MasterOne  |  foo :  my name is  " bar "  and I live in  " foo " .

Master :  MasterTwo  |  foo :  my name is  " bar "  and I live in  " foo " .

Now changing value of  $bar  and printing  $bar  and  $baz
my name is  " baz "  and I live in  " foo " .
my name is  " baz "  and I live in  " foo " .

Now printing again MasterOne and Two
Master :  MasterOne  |  foo :  my name is  " baz "  and I live in  " foo " .

Master :  MasterTwo  |  foo :  my name is  " baz "  and I live in  " foo " .

Now changing MasterTwo ' s foo name and printing again MasterOne and Two
Master: MasterOne | foo: my name is "MasterTwo ' s Foo "  and I live in  " foo " .

Master: MasterTwo | foo: my name is  " MasterTwo ' s Foo" and I live in "foo".

Also printing $bar and $baz
my name is "MasterTwo ' s Foo "  and I live in  " foo " .
my name is  " MasterTwo ' s Foo" and I live in "foo".

 

上个例子解析:

$bar   =   new  foo( ' bar ' );
$m1   =   new  MasterOne(  $bar  );
$m2   =   new  MasterTwo(  $bar  );
 

实例对象$m1与$m2中的$bar是对实例$bar的引用,而非拷贝,这是php5中,对象引用的特点,也就是说
1.$m1或$m2内部,任何对$bar的操作都会影响外部对象实例$bar的相关值。
2.外部对象实例$bar的改变也会影响$m1和$m2内部的$bar的引用相关值。
 

在php4中,要实现如上述的 用一个对象实例去当着另外一个对象的属性时,其等价代码(即引用调用)类似如下:

class  foo{
    var   $bar ;
    function  setBar( & $newBar ){
       $this -> bar  =&  newBar;
   }
}
 

 

5.引用的作用
     如果程序比较大,引用同一个对象的变量比较多,并且希望用完该对象后手工清除它,个人建议用 "&" 方式,然后用$var=null的方式清除. 其它时候还是用php5的默认方式吧. 另外, php5中对于大数组的传递,建议用 "&" 方式, 毕竟节省内存空间使用。


6.取消引用
当你 unset 一个引用,只是断开了变量名和变量内容之间的绑定。这并不意味着变量内容被销毁了。例如:


php
     $a   =   1 ;
     $b   =&   $a ;
     unset  ( $a );
?>  

 


不会 unset $b,只是 $a。


7.global 引用
当用 global $var 声明一个变量时实际上建立了一个到全局变量的引用。也就是说和这样做是相同的:

php
     $var   =&   $GLOBALS [ " var " ];
?>  

 

这意味着,例如,unset $var 不会 unset 全局变量。


 

 

如果在一个函数内部给一个声明为 global 的变量赋于一个引用,该引用只在函数内部可见。可以通过使用 $GLOBALS 数组避免这一点。

Example  在函数内引用全局变量www.2cto.com

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 只改变了本地变量的引用。
 


8.$this
在一个对象的方法中,$this 永远是调用它的对象的引用。


//下面再来个小插曲
php中对于地址的指向(类似指针)功能不是由用户自己来实现的,是由Zend核心实现的,php中引用采用的是“写时拷贝”的原理,就是除非发生写操作,指向同一个地址的变量或者对象是不会被拷贝的。

通俗的讲
1:如果有下面的代码


     $a = " ABC " ;
     $b =& $a ;
?>
 

其实此时 $a与$b都是指向同一内存地址 而并不是$a与$b占用不同的内存

2:如果在上面的代码基础上再加上如下代码

php
   $a = " EFG " ;
?>
 

由于$a与$b所指向的内存的数据要重新写一次了,此时Zend核心会自动判断 自动为$b生产一个$a的数据拷贝,重新申请一块内存进行存储


php的引用(就是在变量或者函数、对象等前面加上&符号)是个高级话题,新手多注意,正确的理解php的引用很重要,对性能有较大影响,而且理解错误可能导致程序错误!

很 多人误解php中的引用跟C当中的指针一样,事实上并非如此,而且很大差别。C语言中的指针除了在数组传递过程中不用显式申明外,其他都需要使用*进行定 义,而php中对于地址的指向(类似指针)功能不是由用户自己来实现的,是由Zend核心实现的,php中引用采用的是“写时拷贝”的原理,就是除非发生 写操作,指向同一个地址的变量或者对象是不会被拷贝的,比如下面的代码:

 

$a   =   array ( ' a ' , ' c ' ... ' n ' );
$b   =   $a ;

如 果程序仅执行到这里,$a和$b是相同的,但是并没有像C那样,$a和$b占用不同的内存空间,而是指向了同一块内存,这就是php和c的差别,并不需要 写成$b=&$a才表示$b指向$a的内存,zend就已经帮你实现了引用,并且zend会非常智能的帮你去判断什么时候该这样处理,什么时候不 该这样处理。

如果在后面继续写如下代码,增加一个函数,通过引用的方式传递参数,并打印输出数组大小。

 

function  printArray( & $arr )  // 引用传递
{
    print ( count ( $arr ));

}
  printArray( $a );

上面的代码中,我们通过引用把$a数组传入printArray()函数,zend引擎会认为printArray()可能会导致对$a的改变,此时就会自动为$b生产一个$a的数据拷贝,重新申请一块内存进行存储。这就是前面提到的“写时拷贝”概念。

如果我们把上面的代码改成下面这样:

 

function  printArray( $arr )    // 值传递
{
    print ( count ( $arr ));
}
   printArray( $a );

上面的代码直接传递$a值到printArray()中,此时并不存在引用传递,所以没有出现写时拷贝。

大家可以测试一下上面两行代码的执行效率,比如外面加入一个循环1000次,看看运行的耗时,结果会让你知道不正确使用引用会导致性能下降30%以上。

自我理解:按传值的话是与函数内的参数无关,相当于局部变量的作用,而按传址(引用)的话却与函数内的参数有关,相当于全局变量的作用.而从性能方面来说,看上面分析就够..


引用做什么
PHP 的引用允许用两个变量来指向同一个内容。意思是,当这样做时:

$a =& $b;
?>
这意味着 $a 和 $b 指向了同一个变量。
Note:

$a 和 $b 在这里是完全相同的,这并不是 $a 指向了 $b 或者相反,而是 $a 和 $b 指向了同一个地方。

Note:

如果具有引用的数组被拷贝,其值不会解除引用。对于数组传值给函数也是如此。

Note:

如果对一个未定义的变量进行引用赋值、引用参数传递或引用返回,则会自动创建该变量。

Example #1 对未定义的变量使用引用

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 以及以后版本):

$bar =& new fooclass();
$foo =& find_var($bar);
?>
自 PHP 5 起, new 自动返回引用,因此在此使用  =& 已经过时了并且会产生 E_STRICT 级别的消息。
Note:

不用 & 运算符导致对象生成了一个拷贝。如果在类中用 $this,它将作用于该类当前的实例。没有用 & 的赋值将拷贝这个实例(例如对象)并且 $this 将作用于这个拷贝上,这并不总是想要的结果。由于性能和内存消耗的问题,通常只想工作在一个实例上面。

尽管可以用 @ 运算符来抑制构造函数中的任何错误信息,例如用 @new,但用 &new 语句时这不起效果。这是 Zend 引擎的一个限制并且会导致一个解析错误。

Warning
如果在一个函数内部给一个声明为 global 的变量赋于一个引用,该引用只在函数内部可见。可以通过使用 $GLOBALS 数组避免这一点。

Example #2 在函数内引用全局变量

$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 语句

$ref = 0;
$row =& $ref;
foreach (array(1, 2, 3) as $row) {
    // do something
}
echo $ref; // 3 - last element of the iterated array
?>
引用做的第二件事是用引用传递变量。这是通过在函数内建立一个本地变量并且该变量在呼叫范围内引用了同一个内容来实现的。例如:

function foo(&$var)
{
    $var++;
}

$a=5;
foo($a);
?>
将使 $a 变成 6。这是因为在 foo 函数中变量 $var 指向了和 $a 指向的同一个内容。更多详细解释见 引用传递。
引用做的第三件事是引用返回
 

 数组引用的一个bug(后来仔细推敲,其实不是bug)


It appears that references can have side-effects.  Below are two examples.  Both are simply copying one array to another.  In the second example, a reference is made to a value in the first array before the copy.  In the first example the value at index 0 points to two separate memory locations. In the second example, the value at index 0 points to the same memory location.

I won't say this is a bug, because I don't know what the designed behavior of PHP is, but I don't think ANY developers would expect this behavior, so look out.

An example of where this could cause problems is if you do an array copy in a script and expect on type of behavior, but then later add a reference to a value in the array earlier in the script, and then find that the array copy behavior has unexpectedly changed.

 

php
//  Example one
$arr1   =   array ( 1 );
echo   " \nbefore:\n " ;
echo   " \$arr1[0] == { $arr1 [0]}\n " ;
$arr2   =   $arr1 ;
$arr2 [ 0 ] ++ ;
echo   " \nafter:\n " ;
echo   " \$arr1[0] == { $arr1 [0]}\n " ;
echo   " \$arr2[0] == { $arr2 [0]}\n " ;


输出:

 

before:
$arr1[0] == 1

after:
$arr1[0] == 1
$arr2[0] == 2
 
 

 

 

// Example two

$arr3=array(1);
$a=&$arr3[0];
echo"\nbefore:\n";
echo"\$a == $a\n";
echo"\$arr3[0] == {$arr3[0]}\n";
$arr4=$arr3;
$arr4[0]++;
echo"\nafter:\n";
echo"\$a == $a\n";
echo"\$arr3[0] == {$arr3[0]}\n";
echo"\$arr4[0] == {$arr4[0]}\n";


 

输出:

 

 

before:
$a == 1
$arr3[0] == 1

after:
$a == 2
$arr3[0] == 2
$arr4[0] == 2
 

 

?>

 


 
 

分析说明:

    对于“Example two”,刚开始还以为是个bug,其实仔细推敲,非也,分析如下,

在赋值(拷贝)

$arr4 = $arr3 ;
 

之前,还有个对$arr3的第一个元素建立引用的过程,即

$a =& $arr3 [ 0 ];
 

所以在后来的赋值拷贝( $arr4=$arr3; ),会把这个引用一并拷贝过去,所以说

$a、$arr3[0]、$arr4[0] 三者其实是引用关系,指向同一个地方。

 

摘自  PPP

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/478314.htmlTechArticlephp的引用(就是在变量或者函数、对象等前面加上符号) 在PHP 中引用的意思是:不同的名字访问同一个变量内容。 与C语言中的指针是有...
声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
php怎么把负数转为正整数php怎么把负数转为正整数Apr 19, 2022 pm 08:59 PM

php把负数转为正整数的方法:1、使用abs()函数将负数转为正数,使用intval()函数对正数取整,转为正整数,语法“intval(abs($number))”;2、利用“~”位运算符将负数取反加一,语法“~$number + 1”。

php怎么实现几秒后执行一个函数php怎么实现几秒后执行一个函数Apr 24, 2022 pm 01:12 PM

实现方法:1、使用“sleep(延迟秒数)”语句,可延迟执行函数若干秒;2、使用“time_nanosleep(延迟秒数,延迟纳秒数)”语句,可延迟执行函数若干秒和纳秒;3、使用“time_sleep_until(time()+7)”语句。

php怎么除以100保留两位小数php怎么除以100保留两位小数Apr 22, 2022 pm 06:23 PM

php除以100保留两位小数的方法:1、利用“/”运算符进行除法运算,语法“数值 / 100”;2、使用“number_format(除法结果, 2)”或“sprintf("%.2f",除法结果)”语句进行四舍五入的处理值,并保留两位小数。

php怎么根据年月日判断是一年的第几天php怎么根据年月日判断是一年的第几天Apr 22, 2022 pm 05:02 PM

判断方法:1、使用“strtotime("年-月-日")”语句将给定的年月日转换为时间戳格式;2、用“date("z",时间戳)+1”语句计算指定时间戳是一年的第几天。date()返回的天数是从0开始计算的,因此真实天数需要在此基础上加1。

php怎么替换nbsp空格符php怎么替换nbsp空格符Apr 24, 2022 pm 02:55 PM

方法:1、用“str_replace(" ","其他字符",$str)”语句,可将nbsp符替换为其他字符;2、用“preg_replace("/(\s|\&nbsp\;||\xc2\xa0)/","其他字符",$str)”语句。

php怎么判断有没有小数点php怎么判断有没有小数点Apr 20, 2022 pm 08:12 PM

php判断有没有小数点的方法:1、使用“strpos(数字字符串,'.')”语法,如果返回小数点在字符串中第一次出现的位置,则有小数点;2、使用“strrpos(数字字符串,'.')”语句,如果返回小数点在字符串中最后一次出现的位置,则有。

php字符串有没有下标php字符串有没有下标Apr 24, 2022 am 11:49 AM

php字符串有下标。在PHP中,下标不仅可以应用于数组和对象,还可应用于字符串,利用字符串的下标和中括号“[]”可以访问指定索引位置的字符,并对该字符进行读写,语法“字符串名[下标值]”;字符串的下标值(索引值)只能是整数类型,起始值为0。

php怎么读取字符串后几个字符php怎么读取字符串后几个字符Apr 22, 2022 pm 08:31 PM

在php中,可以使用substr()函数来读取字符串后几个字符,只需要将该函数的第二个参数设置为负值,第三个参数省略即可;语法为“substr(字符串,-n)”,表示读取从字符串结尾处向前数第n个字符开始,直到字符串结尾的全部字符。

See all articles

热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.能量晶体解释及其做什么(黄色晶体)
2 周前By尊渡假赌尊渡假赌尊渡假赌
仓库:如何复兴队友
4 周前By尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island冒险:如何获得巨型种子
4 周前By尊渡假赌尊渡假赌尊渡假赌

热工具

WebStorm Mac版

WebStorm Mac版

好用的JavaScript开发工具

MinGW - 适用于 Windows 的极简 GNU

MinGW - 适用于 Windows 的极简 GNU

这个项目正在迁移到osdn.net/projects/mingw的过程中,你可以继续在那里关注我们。MinGW:GNU编译器集合(GCC)的本地Windows移植版本,可自由分发的导入库和用于构建本地Windows应用程序的头文件;包括对MSVC运行时的扩展,以支持C99功能。MinGW的所有软件都可以在64位Windows平台上运行。

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

Atom编辑器mac版下载

Atom编辑器mac版下载

最流行的的开源编辑器

SublimeText3 英文版

SublimeText3 英文版

推荐:为Win版本,支持代码提示!