zl程序教程

您现在的位置是:首页 >  后端

当前栏目

浅析php变量作用域的一些问题

PHP变量 问题 一些 浅析 作用域
2023-06-13 09:15:04 时间

昨晚就与到这么一个问题,是全局变量在函数中的问题。今天搜索了一下,发现一篇相当不错的文章,讲了php中的变量作用域。是一位网友翻译的在这贴一下:

变量范围
变量的范围即它定义的上下文背景(译者:说白了,也就是它的生效范围)。大部分的PHP变量只有一个单独的范围。这个单独的范围跨度同样包含了include和require引入的文件。范例:

复制代码代码如下:

<?php
$a=1;
include"b.inc";
?> 

这里变量$a将会在包含文件b.inc中生效。但是,在用户自定义函数中,一个局部函数范围将被引入。任何用于函数内部的变量按缺省情况将被限制在局部函数范围内。范例:
复制代码代码如下:

<?php
$a=1;/*globalscope*/

functionTest()
{
  echo$a;/*referencetolocalscopevariable*/
}

Test();
?> 


这个脚本不会有任何输出,因为echo语句引用了一个局部版本的变量$a,而且在这个范围内,它并没有被赋值。你可能注意到PHP的全局变量和C语言有一点点不同,在C语言中,全局变量在函数中自动生效,除非被局部变量覆盖。这可能引起一些问题,有些人可能漫不经心的改变一个全局变量。PHP中全局变量在函数中使用时必须申明为全局。

Theglobalkeyword
首先,一个使用global的例子:
例子12-1.使用global
复制代码代码如下:
<?php
$a=1;
$b=2;

functionSum()
{
  global$a,$b;

  $b=$a+$b;
}

Sum();
echo$b;
?> 


以上脚本的输出将是"3"。在函数中申明了全局变量$a和$b,任何变量的所有引用变量都会指向到全局变量。对于一个函数能够申明的全局变量的最大个数,PHP没有限制。

在全局范围内访问变量的第二个办法,是用特殊的PHP自定义$GLOBALS数组。前面的例子可以写成:

例子12-2.使用$GLOBALS替代global

复制代码代码如下:
<?php
$a=1;
$b=2;

functionSum()
{
  $GLOBALS["b"]=$GLOBALS["a"]+$GLOBALS["b"];
}

Sum();
echo$b;
?> 


在$GLOBALS数组中,每一个变量为一个元素,键名对应变量名,值变量的内容。$GLOBALS之所以在全局范围内存在,是因为$GLOBALS是一个超全局变量。以下范例显示了超全局变量的用处:

例子12-3.演示超全局变量和作用域的例子

复制代码代码如下:
<?php
functiontest_global()
{
  //大多数的预定义变量并不"super",它们需要用"global"关键字来使它们在函数的本地区域中有效。
  global$HTTP_POST_VARS;

  print$HTTP_POST_VARS["name"];

  //Superglobals在任何范围内都有效,它们并不需要"global"声明。Superglobals是在PHP4.1.0引入的。
  print$_POST["name"];
}
?> 


使用静态变量
变量范围的另一个重要特性是静态变量(staticvariable)。静态变量仅在局部函数域中存在,但当程序执行离开此作用域时,其值并不丢失。看看下面的例子:

例子12-4.演示需要静态变量的例子

复制代码代码如下:
<?php
functionTest()
{
  $a=0;
  echo$a;
  $a++;
}
?> 

本函数没什么用处,因为每次调用时都会将$a的值设为0并输出"0"。将变量加一的$a++没有作用,因为一旦退出本函数则变量$a就不存在了。要写一个不会丢失本次计数值的计数函数,要将变量$a定义为静态的:

例子12-5.使用静态变量的例子

复制代码代码如下:
<?php
functionTest()
{
  static$a=0;
  echo$a;
  $a++;
}
?> 

现在,每次调用Test()函数都会输出$a的值并加一。

静态变量也提供了一种处理递归函数的方法。递归函数是一种调用自己的函数。写递归函数时要小心,因为可能会无穷递归下去。必须确保有充分的方法来中止递归。一下这个简单的函数递归计数到10,使用静态变量$count来判断何时停止:

例子12-6.静态变量与递归函数

复制代码代码如下:
<?php
functionTest()
{
  static$count=0;

  $count++;
  echo$count;
  if($count<10){
  Test();
  }
  $count--;
}
?> 


注:静态变量可以按照上面的例子声明。如果在声明中用表达式的结果对其赋值会导致解析错误。

例子12-7.声明静态变量

复制代码代码如下:
<?php
functionfoo(){
  static$int=0;//correct
  static$int=1+2;//wrong(asitisanexpression)
  static$int=sqrt(121);//wrong(asitisanexpressiontoo)

  $int++;
  echo$int;
}
?> 


全局和静态变量的引用
在Zend引擎1代,驱动了PHP4,对于变量的static和global定义是以references的方式实现的。例如,在一个函数域内部用global语句导入的一个真正的全局变量实际上是建立了一个到全局变量的引用。这有可能导致预料之外的行为,如以下例子所演示的:
复制代码代码如下:
<?php
functiontest_global_ref(){
  global$obj;
  $obj=&newstdclass;
}

functiontest_global_noref(){
  global$obj;
  $obj=newstdclass;
}

test_global_ref();
var_dump($obj);
test_global_noref();
var_dump($obj);
?> 


执行以上例子会导致如下输出:
NULLobject(stdClass)(0){}
类似的行为也适用于static语句。引用并不是静态地存储的:
复制代码代码如下:
<?php
function&get_instance_ref(){
  static$obj;

  echo"Staticobject:";
  var_dump($obj);
  if(!isset($obj)){
  //将一个引用赋值给静态变量
  $obj=&newstdclass;
  }
  $obj->property++;
  return$obj;
}

function&get_instance_noref(){
  static$obj;

  echo"Staticobject:";
  var_dump($obj);
  if(!isset($obj)){
  //将一个对象赋值给静态变量
  $obj=newstdclass;
  }
  $obj->property++;
  return$obj;
}

$obj1=get_instance_ref();
$still_obj1=get_instance_ref();
echo"/n";
$obj2=get_instance_noref();
$still_obj2=get_instance_noref();
?> 


执行以上例子会导致如下输出:
Staticobject:NULLStaticobject:NULLStaticobject:NULLStaticobject:object(stdClass)(1){ ["property"]=> int(1)}

上例演示了当把一个引用赋值给一个静态变量时,第二次调用&get_instance_ref()函数时其值并没有被记住。