ホームページ php教程 php手册 php中引用(变量和函数名前加&符号)用法

php中引用(变量和函数名前加&符号)用法

May 25, 2016 pm 04:48 PM
関数 変数 引用

本文章来给大家介绍在php中我们常看到在在php变量前面加个&符号,这个就是php中引用符号了,它可以用于各种变量、函数、对象了下面我来给各位详细介绍php&符号用法.在 PHP 中引用意味着用不同的名字访问同一个变量内容.这并不像 C 的指针,它们是符号表别名.注意在 PHP 中,变量名和变量内容是不一样的,因此同样的内容可以有不同的名字.PHP的引用是通过在变量名或者函数名前加&符号来实现的.下面解释一下引用的几种用法:

先来看官方法的说明:

引用做什么

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 引擎的一个限制并且会导致一个解析错误.

首先是变量的简单引用,允许你用两个变量来指向同一个内容,举个简单的例子:

<?php
$a = 5; 
$b = &$a; 
echo $b; 
$a++; 
echo $b; 
?>
ログイン後にコピー

运行这段代码是让$b来引用$a的内容,然后改变$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
function &find_var ($param) 
{ 
    /* ...code... */ 
    return $found_var; 
} 
$foo =& find_var ($bar); 
$foo->x = 2; 
?>
ログイン後にコピー

这个例子给$foo 赋值是函数find_var的返回引用,所以在给$foo->x赋值时就是给find_var的返回引用赋值,而不是简单的赋值.PHP引用的最后一个用法是引用定位,主要有两个应用:一个是global 引用,当用 global $var 声明一个变量时实际上建立了一个到全局变量的引用.也就是和$var =& $GLOBALS["var"];是一样的.另外一个是$this的用法,在一个对象的方法中,$this 永远是调用它的对象的引用.与指针的区别引用与指针很像,但是其并不是指针.

看如下的代码:

<?php
$a = 0; 
$b = &a; 
echo $a; //0 
unset($b); 
echo $a; //0
?>
ログイン後にコピー

由于$b只是$a的别名,所以即使$b被释放了,$a没有任何影响,但是指针可不是这样的.

看如下代码:

<?php
#include  
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输出的内容应该一样,事实上结果并非如此.看下面通过引用传递对象的列子:

$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三者内容完全一样,所以可以看出对象并不是按照引用传递,要尽快走出这个误区.


               
               

永久地址:

转载随意~请带上教程地址吧^^

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

golang 関数で新しい関数を動的に作成するためのヒント golang 関数で新しい関数を動的に作成するためのヒント Apr 25, 2024 pm 02:39 PM

Go 言語は、クロージャとリフレクションという 2 つの動的関数作成テクノロジを提供します。クロージャを使用すると、クロージャ スコープ内の変数にアクセスでき、リフレクションでは FuncOf 関数を使用して新しい関数を作成できます。これらのテクノロジーは、HTTP ルーターのカスタマイズ、高度にカスタマイズ可能なシステムの実装、プラグイン可能なコンポーネントの構築に役立ちます。

C++ 関数の命名におけるパラメーターの順序に関する考慮事項 C++ 関数の命名におけるパラメーターの順序に関する考慮事項 Apr 24, 2024 pm 04:21 PM

C++ 関数の名前付けでは、読みやすさを向上させ、エラーを減らし、リファクタリングを容易にするために、パラメーターの順序を考慮することが重要です。一般的なパラメータの順序規則には、アクション-オブジェクト、オブジェクト-アクション、意味論的な意味、および標準ライブラリへの準拠が含まれます。最適な順序は、関数の目的、パラメーターの種類、潜在的な混乱、および言語規約によって異なります。

Java で効率的で保守しやすい関数を記述するにはどうすればよいでしょうか? Java で効率的で保守しやすい関数を記述するにはどうすればよいでしょうか? Apr 24, 2024 am 11:33 AM

効率的で保守しやすい Java 関数を作成するための鍵は、シンプルに保つことです。意味のある名前を付けてください。特殊な状況に対処します。適切な可視性を使用してください。

C++関数のデフォルトパラメータと可変パラメータの長所と短所の比較 C++関数のデフォルトパラメータと可変パラメータの長所と短所の比較 Apr 21, 2024 am 10:21 AM

C++ 関数のデフォルト パラメーターの利点には、呼び出しの簡素化、可読性の向上、エラーの回避などがあります。欠点は、柔軟性が限られていることと、名前の制限があることです。可変引数パラメーターの利点には、無制限の柔軟性と動的バインディングが含まれます。欠点としては、複雑さの増大、暗黙的な型変換、デバッグの難しさなどが挙げられます。

Excel関数の公式の完全なコレクション Excel関数の公式の完全なコレクション May 07, 2024 pm 12:04 PM

1. SUM 関数は、列またはセルのグループ内の数値を合計するために使用されます (例: =SUM(A1:J10))。 2. AVERAGE 関数は、列またはセルのグループ内の数値の平均を計算するために使用されます (例: =AVERAGE(A1:A10))。 3. COUNT 関数。列またはセルのグループ内の数値またはテキストの数をカウントするために使用されます。例: =COUNT(A1:A10)。 4. IF 関数。指定された条件に基づいて論理的な判断を行い、結果を返すために使用されます。対応する結果。

参照型を返す C++ 関数の利点は何ですか? 参照型を返す C++ 関数の利点は何ですか? Apr 20, 2024 pm 09:12 PM

C++ で参照型を返す関数の利点は次のとおりです。 パフォーマンスの向上: 参照による受け渡しによりオブジェクトのコピーが回避され、メモリと時間が節約されます。直接変更: 呼び出し元は、返された参照オブジェクトを再割り当てせずに直接変更できます。コードの簡素化: 参照渡しによりコードが簡素化され、追加の代入操作は必要ありません。

カスタム PHP 関数と定義済み関数の違いは何ですか? カスタム PHP 関数と定義済み関数の違いは何ですか? Apr 22, 2024 pm 02:21 PM

カスタム PHP 関数と定義済み関数の違いは次のとおりです。 スコープ: カスタム関数はその定義のスコープに限定されますが、事前定義関数はスクリプト全体からアクセスできます。定義方法: カスタム関数は function キーワードを使用して定義されますが、事前定義関数は PHP カーネルによって定義されます。パラメータの受け渡し: カスタム関数はパラメータを受け取りますが、事前定義された関数はパラメータを必要としない場合があります。拡張性: カスタム関数は必要に応じて作成できますが、事前定義された関数は組み込みで変更できません。

C++ 関数例外の詳細: カスタマイズされたエラー処理 C++ 関数例外の詳細: カスタマイズされたエラー処理 May 01, 2024 pm 06:39 PM

C++ の例外処理は、特定のエラー メッセージ、コンテキスト情報を提供し、エラーの種類に基づいてカスタム アクションを実行するカスタム例外クラスを通じて強化できます。 std::Exception から継承した例外クラスを定義して、特定のエラー情報を提供します。カスタム例外をスローするには、throw キーワードを使用します。 try-catch ブロックでdynamic_castを使用して、キャッチされた例外をカスタム例外タイプに変換します。実際の場合、open_file 関数は FileNotFoundException 例外をスローします。例外をキャッチして処理すると、より具体的なエラー メッセージが表示されます。

See all articles