## 变量范围
变量的范围即它定义的上下文背景(也就是它的生效范围)。大部分的PHP变量只有一个单独的范围。这个单独的范围跨度同样包含了include和require引入的文件。例如 :
```
<?php
$a = 1;
include 'b.inc';
```
这里变量$a将会在包含文件b.inc中生效。但是,在用户自定义函数中,一个局部函数范围将被引入。任何用于函数内部的变量按缺省情况将被限制在局部函数范围内。例如:
```
<?php
$a = 1; /* global scope */
function Test()
{
echo $a; /* reference to local scope variable 对局部范围变量的引用 */
}
Test();
```
这个脚本不会有任何输出,因为echo语句引用了一个局部版本$a,而且在这个范围内,它并没有被赋值。你可能注意到PHP的全局变量和C语言有点点不同,在C语言中,全局变量在函数中自动生效,除非被局部变量覆盖。这可能引起一些问题,有些人可能不小心就该变了一个全局变量。PHP中全局变量在函数中使用时必须声明为global。
## global 关键字
首先,一个使用global的例子:
**Example #1 使用 global**
```
<?php
$a = 1;
$b = 2;
function Sum()
{
global $a,$b;
$b = $a + $b;
}
Sum();
echo $b;
?>
```
以上脚本的输出将是“3”。在函数中声明了全局变量$a和$b之后,对任一变量的所有引用都会指向其全局版本。对于一个函数能够声明的全局变量的最大个数,PHP没有限制。
在全局范围内访问变量的第二个办法,是用特殊的PHP自定义[$GLOBALS](http://php.net/manual/zh/reserved.variables.globals.php) 数组。前面的例子可以写成:
**Example #2 使用 [$GLOBALS](http://php.net/manual/zh/reserved.variables.globals.php) 替代 global**
```
<?php
$a = 1;
$b = 2;
function Sum()
{
$GLOBALS['b'] = $GLOBALS['a'] + $GLOBALS['b'];
}
Sum();
echo $b;
?>
```
[$GLOBALS](http://php.net/manual/zh/reserved.variables.globals.php) 是一个关联数组,每一个变量为一个元素,键名对应变量名,值对应变量的内容。[$GLOBALS](http://php.net/manual/zh/reserved.variables.globals.php) 之所以在全局范围内存在,是因为 $GLOBALS 是一个[超全局变量](http://php.net/manual/zh/language.variables.superglobals.php)。以下范例显示了超全局变量的用处:
**Example #3 演示超全局变量和作用域的例子**
```
<?php
function test_global()
{
// 大多数的预定义变量并不“super”,它们需要用'global'关键字来使它们在函数的本地区域中有效。
global $HTTP_POST_VARS;
echo $HTTP_POST_VARS['name'];
// Superglobals 在任何范围内都有效,它们并不需要'global'声明。Superglobals 是在 PHP 4.1.0 引入的。
echo $_POST['name'];
}
?>
```
## 使用静态变量
变量范围的另一个重要特性是静态变量(static variable)。静态变量仅在局部函数域中存在,但当程序执行离开此作用域时,其值并不丢失。看看下面的例子:
**Example #4 演示需要静态变量的例子**
```
<?php
function Test()
{
$a = 0;
echo $a;
$a++;
}
```
本函数没有什么用处,因为每次调用时都会将$a的值设为0并输出0。将变量加一的$a++没有作用,因为一旦退出本函数则变量$a就不存在了。要写一个不会丢失本次计数值的计数函数,要将变量$a定义为静态的:
**Example #5 使用静态变量的例子**
```
<?php
functio test()
{
static $a = 0;
echo $a;
$a++;
?>
```
现在,变量$a仅在第一次调用test()函数时被初始化,之后每次调用test()函数都会输出$a的值并加一。
静态变量也提供了一种处理递归函数的方法。递归函数是一种调用自己的函数。写递归函数时要小心,因为可能会无穷递归下去。必须确保有充分的方法来中止递归。以下这个简单的函数递归计数到10,使用静态变量$count来判断何时停止:
**Example #6 静态变量与递归函数**
```
<?php
function test()
{
static $count = 0;
$count++;
echo $count;
if($count < 10){
test();
}
$count--;
}
```
> Note:静态变量可以按照上面的例子声明。如果在声明中用表达式的结果对其赋值会导致解析错误。
**Example #7 声明静态变量**
```
<?php
function foo(){
static $int = 0; // correct
static $int = 1+2; // wrong (as it is an expression)
static $int = sqrt(123); //wrong (as it is an expression too)
$int++;
echo $int;
?>
```
静态声明是在编译时解析的。
> Note:在函数之外使用global关键字不算错。可以用于在一个函数之内包含文件时。
## 全局和静态变量的引用
在Zend引擎1代,它驱动了PHP4,对于变量的 [static](http://php.net/manual/zh/language.variables.scope.php#language.variables.scope.static) 和 [global](http://php.net/manual/zh/language.variables.scope.php#language.variables.scope.global) 定义是以[引用](http://php.net/manual/zh/language.references.php)的方式实现的。例如,在一个函数域内部用global语句导入的一个真正的全局变量实际上是建立了一个到全局变量的引用。这有可能导致预料之外的行为,如以下例子所演示的:
```
<?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);
?>
```
以上例程会输出:
NULL
object(stdClass)(0) {
}
类似的行为也适用于 *static* 语句。引用并不是静态地存储的:
```
<?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: NULL
Static object: NULL
Static object: NULL
Static object: object(stdClass)(1) {
\["property"\]=>
int(1)
}
上例演示了当把一个引用赋值给一个静态变量时,第二次调用 *&get\_instance\_ref()* 函数时其值并没有被*记住*。
- 序言
- 简介
- PHP是什么?
- PHP能做什么?
- 基本语法
- 类型
- boolean(布尔型)
- integer(整型)
- float(浮点型)
- string(字符串)
- array(数组)
- object(对象)
- callable(可调用)
- resource(资源)
- NULL(无类型)
- 伪类型
- 类型转换的判别
- 变量
- 基础
- 预定义变量
- 变量范围
- 可变变量
- 来自PHP之外的变量
- 常量
- 语法
- 魔术常量
- 表达式
- 运算符
- 运算符优先级
- 算术运算符
- 赋值运算符
- 位运算符
- 比较运算符
- 错误控制运算符
- 执行运算符
- 递增/递减运算符
- 逻辑运算符
- 字符串运算符
- 数组运算符
- 类型运算符
- 流程控制
- if
- else
- elseif/else if
- 流程控制的替代语法
- while
- do-whille
- for
- foreach
- break
- continue
- switch
- declare
- return
- require
- include
- require_once
- include_once
- goto
- 函数
- 用户自定义函数
- 函数的参数
- 返回值
- 可变函数
- 内部 (内置)函数
- 匿名函数
- 类与对象
- 简介
- 基本概念
- 属性
- 类的自动加载
- 构造函数
- 访问控制(可见性)