この記事はPHPの引用を包括的に示しており、非常に詳細で、例も付いています。非常に優れた記事であり、必要な友人に参照することをお勧めします。
引用とは
PHP でのクォートとは、同じ変数の内容に異なる名前でアクセスすることを意味します。これは C ポインターとは異なり、参照はシンボル テーブルのエイリアスです。 PHP では変数名と変数の内容が異なるため、同じ内容でも異なる名前が付けられることに注意してください。最も近い類似点は、Unix のファイル名とファイル自体です。変数名はディレクトリ エントリであり、変数の内容はファイル自体です。参照は、Unix ファイル システムのハードリンクと考えることができます。
引用符をどうするか
PHP の参照では、2 つの変数が同じコンテンツを指すことができます。つまり、これを行うとき:
コードは次のとおりです:
これは、$a と $b が同じ変数を指していることを意味します。
注:
$a と $b はここではまったく同じです。$a が $b を指しているわけではなく、その逆でもありません。$a と $b が同じ場所を指しているだけです。
注:
参照を持つ配列がコピーされた場合、その値は逆参照されません。配列値を関数に渡す場合も同様です。
注:
未定義の変数が参照によって割り当てられた場合、参照によって渡された場合、または参照によって返された場合、変数は自動的に作成されます。
例 #1 未定義の変数への参照を使用する
コードは次のとおりです:
同じ構文は、参照を返す関数と new 演算子 (PHP 4.0.4 以降) で使用できます。
コードは次のとおりです:
PHP 5 では、new は自動的に参照を返すため、ここでの =& の使用は廃止され、E_STRICT レベルのメッセージが生成されます。
注:
& 演算子を使用しないと、オブジェクトのコピーが生成されます。クラスで $this を使用すると、そのクラスの現在のインスタンスに適用されます。 & なしで代入するとインスタンス (オブジェクトなど) がコピーされ、そのコピーに $this が適用されますが、必ずしも望ましい結果になるとは限りません。パフォーマンスとメモリ消費の問題のため、通常は 1 つのインスタンスのみで作業する必要があります。
@new などの @ 演算子を使用してコンストラクター内のエラー メッセージを抑制することは可能ですが、これは &new ステートメントを使用する場合には効果がありません。これは Zend エンジンの制限であり、解析エラーが発生します。
警告
関数内でグローバルとして宣言された変数に参照が割り当てられている場合、その参照は関数内でのみ表示されます。これは、$GLOBALS 配列を使用することで回避できます。
例2 関数内でグローバル変数を参照する
コードは次のとおりです:
global $var; は $var =& $GLOBALS['var']; の略称と考えてください。したがって、別の参照を $var に代入しても、ローカル変数への参照が変更されるだけです。
注:
例 #3 引用符と foreach ステートメント
コードは次のとおりです:
参照が行う 2 番目のことは、参照によって変数を渡すことです。これは、関数内にローカル変数を作成し、その変数が呼び出しスコープ内の同じコンテンツを参照することによって実現されます。例:
コードは次のとおりです:
$a=5;
foo($a);
?>
は$aを6に変換します。これは、関数 foo で変数 $var が $a が指すものと同じものを指しているためです。詳細な説明については、「参照による受け渡し」を参照してください。
参照が行う 3 番目のことは、参照を返すことです。
引用とは何ですか
前に述べたように、参照はポインタではありません。これは、次の構造では望ましい効果が得られないことを意味します:
コードは次のとおりです:
これにより、関数の呼び出し時に foo 関数の $var 変数が $bar にバインドされますが、その後 $GLOBALS["baz"] に再バインドされます。関数 foo には変数 $bar が存在しないため ($var として表されますが、$var には変数の内容のみが含まれ、呼び出しはありません)、参照メカニズムを通じて関数呼び出しスコープ内の別の変数に $bar をバインドすることはできません。シンボル テーブルの名前と値のバインディング)。参照リターンを使用して、関数によって選択された変数を参照できます。
参照渡し
関数がその引数の値を変更できるように、参照によって変数を関数に渡すことができます。構文は次のとおりです:
コードは次のとおりです:
関数呼び出しには参照シンボルはなく、関数定義のみにあることに注意してください。パラメーターを参照によって正しく渡すには、関数定義だけで十分です。最近のバージョンの PHP では、foo(&$a); で & を使用すると、「呼び出し時参照渡し」が非推奨であるという警告が表示されます。
以下は参照によって渡すことができます:
foo($a) などの変数
新しいステートメント (foo(new foobar()) など)
関数から返される参照。例:
コードは次のとおりです:
詳細な説明については、リファレンスリターンを参照してください。
他の式は参照渡しできず、結果は未定義です。たとえば、次の参照渡しの例は無効です:
コードは次のとおりです:
これらの条件は、PHP 4.0.4 以降のバージョンで利用できます。
参照による返却
参照リターンは、関数を使用して参照をバインドする必要がある変数を検索する場合に使用します。パフォーマンスを向上させるために戻り参照を使用しないでください。エンジンはそれ自体を最適化するのに十分な機能を備えています。正当な技術的理由がある場合にのみ参照を返してください。参照を返すには、次の構文を使用します:
コードは次のとおりです:
この例では、getValue 関数によって返されるオブジェクトのプロパティには、参照構文がない場合と同様に、コピーされるのではなく値が割り当てられます。
注: パラメーターの受け渡しとは異なり、ここでは両方の場所でアンパサンドを使用する必要があります。これは、通常のコピーではなく参照が返されることを示し、また $myValue が通常の割り当てではなく参照としてバインドされていることを示します。
注: return ($this->value); のような関数から参照を返そうとした場合、参照される変数ではなく式の結果を返そうとしているため、これは機能しません。参照変数を返すことができるのは関数からのみであり、他に方法はありません。コードが動的式または new 演算子の結果を返そうとすると、PHP 4.4.0 および PHP 5.1.0 以降では E_NOTICE エラーが発行されます。
コードは次のとおりです:
$a=test() メソッドで関数を呼び出すと、関数の値が $a に代入されるだけであり、$a への変更は関数内の $b には影響しません。関数を呼び出すと、その関数は $b を返す $b 変数のメモリ アドレスと、$a 変数のメモリ アドレスを同じ場所にポイントすることで、これと同等の効果が生じます ($a=&b;)。 $a の値は $b の値も変更するので、 $a=&test(); を実行すると、$b の値は 5 になります。
引用解除
参照の設定を解除すると、変数名と変数の内容の間のバインドが解除されるだけです。これは、変数の内容が破壊されることを意味するものではありません。例:
コードは次のとおりです:
$b の設定は解除されず、$a のみが解除されます。
これと Unix のリンク解除呼び出しを類推すると、理解が容易になるかもしれません。
引用の位置づけ
多くの PHP 構文構造は参照メカニズムを通じて実装されているため、参照バインディングに関する上記の内容はすべてこれらの構造にも当てはまります。 pass-by-reference や return などの一部の構造については、すでに上で説明しました。参照を使用する他の構造は次のとおりです:
グローバルな引用
global $var で変数を宣言すると、実際にはグローバル変数への参照が作成されます。それはこれを行うのと同じです:
コードは次のとおりです:
これは、たとえば、unset $var ではグローバル変数の設定が解除されないことを意味します。
unset($a) と $a=null を使用した結果は異なります。メモリのブロックに $a のマッピングが 1 つだけある場合、unset($a) は $a=null と同等になり、メモリのブロックに $a のマッピングが 2 つある場合は自動的にリサイクルされます。 $a と $b を組み合わせた場合、 unset($a) を実行すると $a=null が発生し、 $b は変更されず、 $a=null を実行すると $a=$b=null が発生します。
理由: 変数に null を代入すると、その変数に対応するメモリ ブロックの参照カウントが直接 0 に設定され、自動的にリサイクルされます。
$これ
オブジェクト メソッドでは、$this は常に、それを呼び出すオブジェクトへの参照になります。
引用符の役割
プログラムが比較的大きく、同じオブジェクトを参照する変数が多数あり、使用後にオブジェクトを手動でクリアしたい場合は、個人的には "&" メソッドを使用し、その後 $var=null を使用してオブジェクトをクリアすることをお勧めします。それ以外の場合は、デフォルトのメソッドである php5 を使用してください。また、php5 で大きな配列を転送する場合は、メモリ領域を節約できるため、「&」メソッドを使用することをお勧めします。
ここでもう 1 つの小さなエピソードを紹介します。PHP のアドレスのポインティング (ポインタに似た) 関数は、ユーザー自身によって実装されるのではなく、PHP の参照は「コピー オン ライト」の原則を採用しています。つまり、書き込み操作が発生しない限り、同じアドレスを指す変数やオブジェクトはコピーされません。
平たく言うと
1: 次のコードがある場合
コードは次のとおりです:
実際、この時点では、$a と $b が異なるメモリを占有するのではなく、両方とも同じメモリ アドレスを指します。
2: 上記コードに以下のコードを追加すると
コードは次のとおりです:
$a と $b が指すメモリ内のデータは書き換える必要があるため、Zend コアはこの時点で $b 用の $a のデータ コピーを自動的に生成し、メモリの一部を再適用することを自動的に決定します。保管用に。
上記はすべて PHP の引用に関するものです。皆さんに気に入っていただければ幸いです。