이 기사는 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 값도 변경된다는 의미입니다.
변수 범위를 피하는 또 다른 방법은 수퍼전역 변수($_GET, $_POST, $_REQUEST 등)를 활용하는 것입니다. 이러한 변수는 함수 내에서 자동으로 액세스할 수 있습니다(따라서 슈퍼전역 변수입니다). 함수 내에서 사용할 수 있도록 $GLOBALS 배열에 요소를 추가할 수도 있습니다.
즉, 함수 내에서는 전역 변수를 사용하지 않는 것이 가장 좋습니다. 함수를 설계할 때 필요에 따라 모든 값을 매개변수로 받아들이고 필요에 따라 값을 반환하도록 해야 합니다. 함수 내에서 전역 변수에 의존하면 함수가 상황에 더 의존하게 되어 유용성이 떨어집니다.
PHP의 변수에는 주로 내장된 슈퍼 전역 변수, 일반 변수, 상수, 전역 변수, 정적 변수 등이 포함됩니다.
내장된 슈퍼 전역 변수는 스크립트의 어느 곳에서나 사용하고 볼 수 있습니다. 즉, PHP 페이지의 값 중 하나를 변경하면 다른 PHP 페이지에서 사용될 때 해당 값도 변경됩니다.
•한 번 선언된 상수는 전역적으로 표시됩니다. 즉, 함수 내부 및 외부에서 사용할 수 있지만 이는 한 페이지에 포함된 PHP 스크립트(include 및 include_once를 통해 포함하는 PHP 스크립트 포함)로만 제한됩니다. 다른 페이지는 더 이상 사용할 수 없습니다.
•스크립트에 선언된 전역 변수는 스크립트 전체에서 볼 수 있지만 함수 내부에서는 볼 수 없습니다. 함수 내부 변수의 이름이 전역 변수와 동일한 경우 함수 내부 변수가 우선합니다.
•함수 내부에서 사용되는 변수를 전역 변수로 선언할 때 그 이름은 전역 변수의 이름과 일치해야 합니다. 이 경우 함수 내에서 함수 외부의 전역 변수를 사용할 수 있습니다. 함수 내부의 변수 이름이 외부 전역 변수와 동일하기 때문에 외부 변수를 덮어쓰는 상황을 피할 수 있습니다.
•함수 내부에서 정적으로 생성되고 선언된 변수는 함수 외부에서 볼 수 없지만 함수를 여러 번 실행하는 동안 값이 유지될 수 있습니다. 가장 일반적인 경우는 함수를 재귀적으로 실행하는 동안입니다.
•함수 내부에 생성된 변수는 함수에 대해 로컬이며 함수가 종료되면 더 이상 존재하지 않습니다.
수퍼 전역 변수의 전체 목록은 다음과 같습니다.
•.$GOBALS 모든 전역 변수 배열
•.$_SERVER 서버 환경 변수 배열
•.$_POST 다음을 통해 이 스크립트에 전달된 변수 배열 POST 메소드
•.$ _GET GET 메소드를 통해 스크립트에 전달된 변수 배열
•.$_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 중국어 웹사이트의 기타 관련 기사를 참조하세요!