PHPリファレンス(&)の詳しい説明と注意事項
PHP リファレンス (つまり、変数 または 関数 または オブジェクト または オブジェクト メソッド など & 記号)
PHP での引用とは、異なる名前が同じ変数の内容にアクセスすることを意味します。C言語のポインタとは異なります。 C言語におけるポインタは、変数の内容とメモリ上に格納されているアドレスを格納します。
1. 変数参照
PHP の参照を使用すると、2 つの変数を使用して同じコンテンツを指すことができます。
<? $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 "<br>"; echo $b; //输出101 ?>
注:
パラメータを渡したい場合は、上記の "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() 関数を呼び出すとき、その関数は $b を返す $b 変数のメモリ アドレスと、$a 変数のメモリ アドレスを同じ場所にポイントすることで、同等の結果が生成されます。この効果 ($a = &$b;) したがって、$a の値を変更すると、$b の値も変更されるため、 を実行した後、
その後、$b の値は 5 になります。$a = &test(); $a = 5;
ここで静的変数を使用しているのは、関数の参照戻り値を誰もが理解できるようにするためです。実際、関数の参照戻り値は主にオブジェクトで使用されます。
公式の PHP サンプルを添付します:
<?php // This is the way how we use pointer to access variable inside the class. 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. オブジェクト参照
上記のコードは、PHP5 で実行した場合の結果です。<?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 におけるオブジェクトの割り当ては参照プロセスです。上記の列の $b = new a; $c = $b; は、実際には $b = new a; $c = &$b; と同じですが、場合によってはオブジェクトを参照します。オブジェクトのコピーを作成したいと考えており、元のオブジェクトへの変更がコピーに影響しないことを望んでいる場合があります。このような目的のために、PHP5 では __clone と呼ばれる特別なメソッドが定義されています。
PHP 5 では、new は自動的に参照を返すため、ここでの =& の使用は廃止され、E_STRICT レベルのメッセージが生成されます。 PHP4 では、オブジェクトの割り当ては $b = new a のようなコピー プロセスです。ここで、new 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__ . '".' . " "; } function setName($str) { $this->name = $str; } } class MasterOne { protected $foo; function __construct($f) { $this->foo = $f; } function __toString() { return 'Master: ' . __CLASS__ . ' foo: ' . $this->foo . " "; } 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 . " "; } function setFooName($str) { $this->foo->setName($str); } } $bar = new foo('bar'); print(" "); print("Only Created $bar and printing $bar "); print($bar); print(" "); print("Now $baz is referenced to $bar and printing $bar and $baz "); $baz = &$bar; print($bar); print(" "); print("Now Creating MasterOne and Two and passing $bar to both constructors"); $m1 = new MasterOne($bar); $m2 = new MasterTwo($bar); print($m1); print($m2); print(" "); print("Now changing value of $bar and printing $bar and $baz"); $bar->setName('baz'); print($bar); print($baz); print(" "); print("Now printing again MasterOne and Two"); print($m1); print($m2); print(" "); print("Now changing MasterTwo's foo name and printing again MasterOne and Two"); $m2->setFooName('MasterTwo\'s Foo'); print($m1); print($m2); print("Also printing $bar and $baz"); 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);
实例对象$m1与$m2中的$bar是对实例$bar的引用,而非拷贝,这是PHP5中,对象引用的特点,也就是说
- $m1或$m2内部,任何对$bar的操作都会影响外部对象实例$bar的相关值。
- 外部对象实例$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 在函数内引用全局变量
<?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' "; // var2 is set to '' global_references(true); echo "var2 is set to '$var2' "; // 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安全编程注意事项
- 网站空间安全注意事项
- 在php中分别使用curl的post提交数据的方法和get获取网页数据的方法总结
- php递归函数中使用return需注意
- php利用array_flip实现数组键值交换去除数组重复值
- php利用正则过滤各种标签,空格,换行符的代码
- ThinkPHP内置模板引擎的使用方法总结
- PHP 利用 Curl Functions 实现多线程抓取网页和下载文件

ホットAIツール

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

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

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

Video Face Swap
完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

人気の記事

ホットツール

メモ帳++7.3.1
使いやすく無料のコードエディター

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

ゼンドスタジオ 13.0.1
強力な PHP 統合開発環境

ドリームウィーバー CS6
ビジュアル Web 開発ツール

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

ホットトピック











Win11XboxGameBar を完全にアンインストールするにはどうすればよいですか? Xbox GameBar は、システムに付属するゲーム プラットフォームです。ゲームの録画、スクリーンショット、ソーシャル機能のためのツールを提供します。ただし、大量のメモリを消費するため、アンインストールするのは簡単ではありません。アンインストールしたい友達もいますが、完全にアンインストールする方法を以下に紹介します。方法 1. Windows ターミナルを使用する 1. [Win+X] キーの組み合わせを押すか、タスクバーの [Windows スタート メニュー] を [右クリック] して、表示されるメニュー項目から [ターミナル アドミニストレータ] を選択します。 2. [ユーザー アカウント制御] ウィンドウで、このアプリがデバイスに変更を加えることを許可しますか? [はい]をクリックします。 3. 次のコマンドを実行します: Get-AppxP

ファンクションとは、関数を意味します。これは、特定の関数を備えた再利用可能なコード ブロックです。プログラムの基本コンポーネントの 1 つです。入力パラメータを受け取り、特定の操作を実行し、結果を返すことができます。その目的は、再利用可能なコード ブロックをカプセル化することです。コードの再利用性と保守性を向上させるコード。

Safari アイコンを使用して MacTouchBar の「TouchID ログイン」問題を修正する Touch Bar を強制的に再起動すると、問題が修正されます。Mac で /Applications/Utilities フォルダにある ActivityMonitor を開くか、Command+Spacebar を押して Spotlight を使用できます。 「ActivityMonitor」と入力し、戻って起動します。 アクティビティ モニターの検索機能を使用して、「Touch」を検索します。 「TouchBarServer」を選択し、アクティビティ モニター ツールバーの (X) 終了ボタンをクリックします。 「強制終了」を選択して強制的に終了します。

この記事では、Python の enumerate() 関数と「enumerate()」関数の目的について学びます。 enumerate() 関数とは何ですか? Python の enumerate() 関数は、データ コレクションをパラメータとして受け取り、列挙オブジェクトを返します。列挙オブジェクトはキーと値のペアとして返されます。キーは各項目に対応するインデックス、値は項目です。構文 enumerate(iterable,start) パラメータ iterable - 渡されたデータ コレクションは、iterablestart と呼ばれる列挙オブジェクトとして返すことができます。 - 名前が示すように、列挙オブジェクトの開始インデックスは start によって定義されます。無視したら

MySQL.proc テーブルの役割と機能の詳細な説明。MySQL は人気のあるリレーショナル データベース管理システムです。開発者が MySQL を使用する場合、多くの場合、ストアド プロシージャ (StoredProcedure) の作成と管理が必要になります。 MySQL.proc テーブルは非常に重要なシステム テーブルであり、ストアド プロシージャの名前、定義、パラメータなど、データベース内のすべてのストアド プロシージャに関連する情報が保存されます。この記事では、MySQL.proc テーブルの役割と機能について詳しく説明します。

Vue の使い方と機能.use Function Vue は、多くの便利な機能を提供する人気のフロントエンド フレームワークです。その 1 つは Vue.use 関数で、これを使用すると Vue アプリケーションでプラグインを使用できるようになります。この記事では、Vue.use 関数の使い方と機能を紹介し、いくつかのコード例を示します。 Vue.use 関数の基本的な使用法は非常に簡単です。Vue がインスタンス化される前に関数を呼び出し、使用するプラグインをパラメータとして渡すだけです。簡単な例を次に示します。 // プラグインの導入と使用

clearstatcache() 関数は、ファイル ステータス キャッシュをクリアするために使用されます。 PHP は、次の関数によって返された情報をキャッシュします。 -stat()lstat()file_exists()is_writable()is_readable()is_executable()is_file()is_dir()filegroup()fileowner()filesize()filetype()fileperms()より良いパフォーマンスを提供するため。構文 voidclearstatecache() パラメータ NA 戻り値 clearstatcache(

file_exists メソッドは、ファイルまたはディレクトリが存在するかどうかを確認します。チェックするファイルまたはディレクトリのパスを引数として受け取ります。これは、ファイルを処理する前に存在するかどうかを知る必要がある場合に役立ちます。これにより、新しいファイルを作成するときに、この関数を使用してファイルがすでに存在するかどうかを確認できます。構文 file_exists($file_path) パラメータ file_path - 存在をチェックするファイルまたはディレクトリのパスを設定します。必須。 return file_exists() メソッドが戻ります。ファイルまたはディレクトリが存在する場合は TrueFalse を返し、ファイルまたはディレクトリが存在しない場合は TrueFalse を返します。 例として、「candidate.txt」ファイルのチェックを見てみましょう。
