引用返回

引用返回用在当想用函数找到引用应该被绑定在哪一个变量上面时。不要用返回引用来增加性能,引擎足够聪明来自己进行优化。仅在有合理的技术原因时才返回引用!要返回引用,使用此语法:

<?php
function &find_var($param)
{
    
/* ...code... */
    
return $found_var;
}

$foo =& find_var($bar);
$foo->x = 2;
?>

本例中 find_var 函数所返回的对象的属性将被赋值,而不是拷贝,就和没有用引用语法一样。

注: 和参数传递不同,这里必须在两个地方都用 & 符号――指出返回的是一个引用,而不是通常的一个拷贝,同样也指出 $foo 是作为引用的绑定,而不是通常的赋值。

注: 如果试图这样从函数返回引用:return ($found_var);,这将不会起作用,因为在试图返回一个表达式的结果而不是一个引用的变量。只能从函数返回引用变量――没别的方法。如果代码试图返回一个动态表达式或 new 运算符的结果,自 PHP 4.4.0 和 PHP 5.1.0 起会发出一条 E_NOTICE 错误。


add a note add a note User Contributed Notes
neozenkai at yahoo dot com
16-Jul-2006 04:38
While trying to create a function to return Database connection objects, it took me a while to get this right:

<?php

class TestClass
{
   var
$thisVar = 0;

   function
TestClass($value)
   {
      
$this->thisVar = $value;
   }

   function &
getTestClass($value)
   {
       static
$classes;

       if (!isset(
$classes[$value]))
       {
          
$classes[$value] = new TestClass($value);
       }

       return
$classes[$value];
   }
}

echo
"<pre>";

echo
"Getting class1 with a value of 432\n";
$class1 =& TestClass::getTestClass(432);
echo
"Value is: " . $class1->thisVar . "\n";

echo
"Getting class2 with a value of 342\n";
$class2 =& TestClass::getTestClass(342);
echo
"Value is: " . $class2->thisVar . "\n";

echo
"Getting class3 with the same value of 432\n";
$class3 =& TestClass::getTestClass(432);
echo
"Value is: " . $class3->thisVar . "\n";

echo
"Changing the value of class1 to 3425, which should also change class3\n";
$class1->thisVar = 3425;

echo
"Now checking value of class3: " . $class3->thisVar . "\n";

?>

Which outputs:

Getting class1 with a value of 432
Value is: 432
Getting class2 with a value of 342
Value is: 342
Getting class3 with the same value of 432
Value is: 432
Changing the value of class1 to 3425, which should also change class3
Now checking value of class3: 3425

Note that PHP syntax is different from C/C++ in that you must use the & operator in BOTH places, as stated by the manual. It took me a while to figure this out.
rwruck
28-Feb-2006 01:22
The note about using parentheses when returning references is only true if the variable you try to return does not already contain a reference.

<?php
// Will return a reference
function& getref1()
  {
 
$ref =& $GLOBALS['somevar'];
  return (
$ref);
  }

// Will return a value (and emit a notice)
function& getref2()
  {
 
$ref = 42;
  return (
$ref);
  }

// Will return a reference
function& getref3()
  {
  static
$ref = 42;
  return (
$ref);
  }
?>
warhog at warhog dot net
13-Dec-2005 04:04
firstly a note on the post below: technically correct that -> "to get a reference to an exsisting class and it's properties" should be "...to an existing object..."

in PHP5 it's senseless to return objects by reference.. let's say you have, as in the post below, a class which should return a reference to its own instance (maybe it's been created using the singleton pattern..), than it's no problem to simply return that variable holding the instance.
In PHP5 variables holding objects are always references to a specific object, so when you return (=copy) a variable "having" an object you in fact return a reference to that object.

Because of that behaviour, which is very common for many programming languages, especially those, which are to be compiled (due to memory problems and so on), you have to use the clone-function which was introduced in PHP5.

Here an example to underline what i mean:

<?php

class sample_singleton
{
 
protected static $instance = NULL;
 
 
private function __construct()
  { }

 
public static function create()
  {
self::$instance = new sample_singleton(); }

 
public static function getInstance()
  { return
self::$instance; }

 
public function yea()
  { echo
"wuow"; }
}

sample_singleton::create();

// $singleton will be a reference to sample_singleton::$instance
$singleton = sample_singleton::getInstance();

$singleton->yea();

?>

Note some more (maybe) strange behaviour: although $instance is private you can have a reference pointing on it. Maybe that does not seem strange to you in any way, but maybe you wondered as i did : )

The code posted here was just a sample to illustrate my posting, for using the singleton pattern please look it up in the manual. I just used this cause it was the simplest example which went in my mind right know.
willem at designhulp dot nl
09-Oct-2005 07:54
There is an important difference between php5 and php4 with references.

Lets say you have a class with a method called 'get_instance' to get a reference to an exsisting class and it's properties.

<?php
class mysql {
   function
get_instance(){
      
// check if object exsists
      
if(empty($_ENV['instances']['mysql'])){
          
// no object yet, create an object
          
$_ENV['instances']['mysql'] = new mysql;
       }
      
// return reference to object
      
$ref = &$_ENV['instances']['mysql'];
       return
$ref;
   }
}
?>

Now to get the exsisting object you can use
mysql::get_instance();

Though this works in php4 and in php5, but in php4 all data will be lost as if it is a new object while in php5 all properties in the object remain.
obscvresovl at NOSPAM dot hotmail dot com
25-Dec-2004 05:09
An example of returning references:

<?

$var
= 1;
$num = NULL;

function &
blah()
{
  
$var =& $GLOBALS["var"]; # the same as global $var;
  
$var++;
   return
$var;
}

$num = &blah();

echo
$num; # 2

blah();

echo
$num; # 3

?>

Note: if you take the & off from the function, the second echo will be 2, because without & the var $num contains its returning value and not its returning reference.
hawcue at yahoo dot com
17-Mar-2004 11:58
Be careful when using tinary operation condition?value1:value2

See the following code:

$a=1;
function &foo()
{
  global $a;
  return isset($a)?$a:null;
}
$b=&foo();
echo $b;  // shows 1
$b=2;
echo $a;  // shows 1 (not 2! because $b got a copy of $a)

To let $b be a reference to $a, use "if..then.." in the function.
contact at infopol dot fr
13-Feb-2004 01:36
A note about returning references embedded in non-reference arrays :

<?
$foo
;

function
bar () {
   global
$foo;
  
$return = array();
  
$return[] =& $foo;
   return
$return;
}

$foo = 1;
$foobar = bar();
$foobar[0] = 2;
echo
$foo;
?>

results in "2" because the reference is copied (pretty neat).
zayfod at yahoo dot com
04-Dec-2003 01:23
There is a small exception to the note on this page of the documentation. You do not have to use & to indicate that reference binding should be done when you assign to a value passed by reference the result of a function which returns by reference.

Consider the following two exaples:

<?php

function    & func_b ()
{
  
$some_var = 2;
   return
$some_var;
}

function   
func_a (& $param)
{
  
# $param is 1 here
  
$param = & func_b();
  
# $param is 2 here
}

$var = 1;
func_a($var);
# $var is still 1 here!!!

?>

The second example works as intended:

<?php

function    & func_b ()
{
  
$some_var = 2;
   return
$some_var;
}

function   
func_a (& $param)
{
  
# $param is 1 here
  
$param = func_b();
  
# $param is 2 here
}

$var = 1;
func_a($var);
# $var is 2 here as intended

?>

(Experienced with PHP 4.3.0)