この記事は、PHP 変数スコープの詳細な分析と紹介です。必要な方は参照してください。
PHP のすべての変数にはスコープがあります。つまり、変数にはフィールドにアクセスできる (つまり、その値にアクセスできます)。 。まず、変数のスコープは変数が存在するページです。したがって、$var を定義すると、ページの残りの部分は $var にアクセスできますが、他のページは通常 (特別な変数が使用されない限り) $var にアクセスできません。
インクルード ファイルは元の (インクルードされた) スクリプトの一部であるかのように動作するため、include() 行の前に定義された変数はインクルードされたファイルで使用できます。さらに、インクルード ファイル内で定義された変数は、include() 行以降の親 (インクルード) スクリプトで使用できます。
独自に定義した関数を使用すると、これらすべてがあまり目立たなくなります。これらの関数には独自のスコープがあります。つまり、関数内で使用される変数は関数の外では使用できず、関数の外で定義された変数は関数内では使用できません。このため、関数内の変数は関数の外側の変数と同じ名前を持つことができますが、それでも値が異なる完全に異なる変数です。これは、ほとんどの初心者プログラマーにとってわかりにくい概念です。
関数内の変数のスコープを変更するには、global ステートメントを使用できます。
コードは次のとおりです:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php function function_name() { global $var; } $var=20; function_name(); // Function call. ?>
この例では、関数内の $var は関数外の $var と同じになります。これは、変数 $var の値がすでに 20 であり、この値が関数内で変更されると、関数外の $var の値も変更されることを意味します。
変数スコープを回避するもう 1 つの方法は、スーパーグローバル変数: $_GET、$_POST、$_REQUEST などを利用することです。これらの変数は関数内で自動的にアクセスできます (したがって、これらはスーパーグローバル変数です)。 $GLOBALS 配列に要素を追加して、関数内で使用できるようにすることもできます。
言い換えれば、関数内でグローバル変数を使用しないことが最善です。関数を設計するときは、必要に応じてすべての値をパラメーターとして受け入れ、必要に応じて任意の値を返すようにする必要があります。関数内でグローバル変数に依存すると、コンテキスト依存性が高まるため、有用性が低くなります。
PHP の変数には主に、組み込みスーパー グローバル変数、一般変数、定数、グローバル変数、静的変数などが含まれます。
組み込みスーパー グローバル変数は、スクリプト内のどこでも使用および表示できます。つまり、PHP ページ内の値の 1 つを変更すると、他の PHP ページで使用されたときにその値も変更されます。
•一度宣言された定数はグローバルに参照可能になります。つまり、関数の内部と外部で使用できますが、これは 1 つのページに含まれる PHP スクリプト (include および include_once で含めた PHP スクリプトを含む) にのみ限定されます。他のページは使用できなくなります。
•スクリプト内で宣言されたグローバル変数はスクリプト全体で参照できますが、関数内では参照できません。関数内の変数がグローバル変数と同じ名前である場合、関数内の変数が優先されます。
•関数内で使用される変数がグローバル変数として宣言されている場合、その名前はグローバル変数の名前と一致している必要があります。この場合、関数内でグローバル変数を関数の外で使用することができます。関数内の変数が外部グローバル変数と同じ名前を持つため、外部変数が上書きされる状況を回避できます。 •関数内で作成され、static として宣言された変数は、関数の外では見ることができませんが、関数の複数回の実行中に値を維持できます。最も一般的なケースは、関数の再帰実行中です。
•関数内で作成された変数は関数に対してローカルであり、関数が終了すると存在しなくなります。
スーパーグローバル変数の完全なリストは次のとおりです: •.$GOBALS すべてのグローバル変数の配列
•.$_SERVER サーバー環境変数の配列
•.$_POST 経由でこのスクリプトに渡される変数の配列POST メソッド
•.$ _GET GET メソッド経由でスクリプトに渡される変数の配列
•.$_COOKIE Cookie 変数の配列
•.$_FILES ファイルのアップロードに関連する変数の配列
•.$_ENV 環境変数の配列
• .$_REQUEST すべてのユーザーが入力した変数の配列 $_GET $_POST $_COOKIE
•.$_SESSION セッション変数配列
インスタンスの説明:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php $a = 4; function sendValue($x) { echo $x; } sendValue($a); ?>説明: $a は関数の外で定義されており、関数はパラメーターを定義します。関数が呼び出されるとき、$a はパラメーターとして渡されます。したがって、上記のコードは正常に実行できます。
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php $a = 4; function sendValue() { echo $a; } sendValue(); ?>
讲解:当函数被调用时,$a不能以参数的形式被传递。所以上面代码不能够正常运行。
变量范围
变量的范围即它定义的上下文背景(译者:说白了,也就是它的生效范围)。大部分的 PHP 变量只有一个单独的范围。这个单独的范围跨度同样包含了 include 和 require 引入的文件。范例:
代码如下:
<?php $a = 1; include "b.inc"; ?>
这里变量 $a 将会在包含文件 b.inc 中生效。但是,在用户自定义函数中,一个局部函数范围将被引入。任何用于函数内部的变量按缺省情况将被限制在局部函数范围内。范例:
代码如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php $a = 1; /* global scope */ function Test() { echo $a; /* reference to local scope variable */ } Test(); ?>
这个脚本不会有任何输出,因为 echo 语句引用了一个局部版本的变量 $a,而且在这个范围内,它并没有被赋值。你可能注意到 PHP 的全局变量和 C 语言有一点点不同,在 C 语言中,全局变量在函数中自动生效,除非被局部变量覆盖。这可能引起一些问题,有些人可能漫不经心的改变一个全局变量。PHP 中全局变量在函数中使用时必须申明为全局。
The global keyword
首先,一个使用 global 的例子:
例子 12-1. 使用 global
代码如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php $a = 1; $b = 2; function Sum() { global $a, $b; $b = $a + $b; } Sum(); echo $b; ?>
以上脚本的输出将是 "3"。在函数中申明了全局变量 $a 和 $b,任何变量的所有引用变量都会指向到全局变量。对于一个函数能够申明的全局变量的最大个数,PHP 没有限制。
在全局范围内访问变量的第二个办法,是用特殊的 PHP 自定义 $GLOBALS 数组。前面的例子可以写成:
例子 12-2. 使用 $GLOBALS 替代 global
代码如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php $a = 1; $b = 2; function Sum() { $GLOBALS["b"] = $GLOBALS["a"] + $GLOBALS["b"]; } Sum(); echo $b; ?>
在 $GLOBALS 数组中,每一个变量为一个元素,键名对应变量名,值变量的内容。$GLOBALS 之所以在全局范围内存在,是因为 $GLOBALS 是一个超全局变量。以下范例显示了超全局变量的用处:
例子 12-3. 演示超全局变量和作用域的例子
代码如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php function test_global() { // 大多数的 预定义变量 并不 "super",它们需要用 'global' 关键字来使它们在函数的本地区域中有效。 global $HTTP_POST_VARS; print $HTTP_POST_VARS['name']; // Superglobals 在任何范围内都有效,它们并不需要 'global' 声明。Superglobals 是在 PHP 4.1.0 引入的。 print $_POST['name']; } ?>
使用静态变量
变量范围的另一个重要特性是静态变量(static variable)。静态变量仅在局部函数域中存在,但当程序执行离开此作用域时,其值并不丢失。看看下面的例子:
例子 12-4. 演示需要静态变量的例子
代码如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php function Test () { $a = 0; echo $a; $a++; } ?>
本函数没什么用处,因为每次调用时都会将 $a 的值设为 0 并输出 "0"。将变量加一的 $a++ 没有作用,因为一旦退出本函数则变量 $a 就不存在了。要写一个不会丢失本次计数值的计数函数,要将变量 $a 定义为静态的:
例子 12-5. 使用静态变量的例子
代码如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php function Test() { static $a = 0; echo $a; $a++; } ?>
现在,每次调用 Test() 函数都会输出 $a 的值并加一。
静态变量也提供了一种处理递归函数的方法。递归函数是一种调用自己的函数。写递归函数时要小心,因为可能会无穷递归下去。必须确保有充分的方法来中止递归。一下这个简单的函数递归计数到 10,使用静态变量 $count 来判断何时停止:
例子 12-6. 静态变量与递归函数
代码如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php function Test() { static $count = 0; $count++; echo $count; if ($count < 10) { Test (); } $count--; } ?>
注: 静态变量可以按照上面的例子声明。如果在声明中用表达式的结果对其赋值会导致解析错误。
例子 12-7. 声明静态变量
代码如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php function foo(){ static $int = 0; // correct static $int = 1+2; // wrong (as it is an expression) static $int = sqrt(121); // wrong (as it is an expression too) $int++; echo $int; } ?>
全局和静态变量的引用
在 Zend 引擎 1 代,驱动了 PHP4,对于变量的 static 和 global 定义是以 references 的方式实现的。例如,在一个函数域内部用 global 语句导入的一个真正的全局变量实际上是建立了一个到全局变量的引用。这有可能导致预料之外的行为,如以下例子所演示的:
代码如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php function test_global_ref() { global $obj; $obj = &new stdclass; } function test_global_noref() { global $obj; $obj = new stdclass; } test_global_ref(); var_dump($obj); test_global_noref(); var_dump($obj); ?>
执行以上例子会导致如下输出:
代码如下:
NULLobject(stdClass)(0) {}
类似的行为也适用于 static 语句。引用并不是静态地存储的:
代码如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php function &get_instance_ref() { static $obj; echo "Static object: "; var_dump($obj); if (!isset($obj)) { // 将一个引用赋值给静态变量 $obj = &new stdclass; } $obj->property++; return $obj; } function &get_instance_noref() { static $obj; echo "Static object: "; var_dump($obj); if (!isset($obj)) { // 将一个对象赋值给静态变量 $obj = new stdclass; } $obj->property++; return $obj; } $obj1 = get_instance_ref(); $still_obj1 = get_instance_ref(); echo "\n"; $obj2 = get_instance_noref(); $still_obj2 = get_instance_noref(); ?>
执行以上例子会导致如下输出:
代码如下:
Static object: NULLStatic object: NULLStatic object: NULLStatic object: object(stdClass)(1) { ["property"]=> int(1)}
上例演示了当把一个引用赋值给一个静态变量时,第二次调用 &get_instance_ref() 函数时其值并没有被记住。
以上がPHPの変数スコープとは何ですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。