`

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

    博客分类:
  • PHP
阅读更多
学过C的人用php的时候一般会相当顺手,而且感到php太方便太轻松。但在变量作用域这方面却与c有不同的地方,搞不好会相当郁闷,就找不到错误所在。

在php编程中,遇到一个全局变量在函数中的问题。
这里分享下php 变量作用域的一篇文章,有需要的朋友参考下。

变量范围
变量的范围即它定义的上下文背景(译者:说白了,也就是它的生效范围)。大部分的 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 中全局变量在函数中使用时必须申明为全局。

The global keyword
首先,一个使用 global 的例子:
例子 12-1. 使用 global
<?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
<?php
$a = 1;
$b = 2;

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

Sum();
echo $b;
?> 


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

例子 12-3. 演示超全局变量和作用域的例子
<?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. 演示需要静态变量的例子
<?php
function Test ()
{
   $a = 0;
   echo $a;
   $a++;
}
?> 


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

例子 12-5. 使用静态变量的例子
<?php
function Test()
{
   static $a = 0;
   echo $a;
   $a++;
}
?> 


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

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

例子 12-6. 静态变量与递归函数
<?php
function Test()
{
   static $count = 0;

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


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

例子 12-7. 声明静态变量
<?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 语句导入的一个真正的全局变量实际上是建立了一个到全局变量的引用。这有可能导致预料之外的行为,如以下例子所演示的:
<?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 语句。引用并不是静态地存储的:
<?php
function &get_instance_ref() {
   static $obj;

   echo "Static object: ";
   var_dump($obj);
   if (!isset($obj)) {
   // 将一个引用赋值给静态变量
   //by [url=http://www.jbxue.com]www.jbxue.com[/url]
   $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() 函数时其值并没有被记住。
分享到:
评论

相关推荐

    深入浅析JavaScript中的作用域和上下文

    3. **块级作用域**:在ES6之前,JavaScript没有块级作用域,但ES6引入了`let`关键字,使得变量能够在特定代码块(如`if`、`for`或`while`语句内)中定义并保持其作用域限制。 - 示例: ```javascript function ...

    深化浅析JavaScript中的作用域和上下文_.docx

    在ES6之前,JavaScript没有块级作用域,但ES6引入了`let`关键字,允许在块级结构(如`if`、`for`循环等)中定义变量,这些变量仅在其所在的块内可见,解决了`var`可能导致的一些问题,如变量提升。 **函数作用域**...

    浅析JavaScript中作用域和作用域链

    JavaScript中的作用域是编程语言中控制变量访问范围的规则,它决定了变量在何处可以被访问。在JavaScript中,主要有两种作用域:全局作用域和局部作用域。 1. **全局作用域**:全局作用域指的是在函数外部定义的...

    浅析php中常量,变量的作用域和生存周期

    在PHP编程语言中,理解和掌握变量、常量以及它们的作用域和生存周期是至关重要的。本文将深入浅出地分析这些概念。 首先,PHP中的变量是动态类型,意味着你无需在声明时指定变量的类型。变量以"$"符号开头,如 `$...

    浅析js封装和作用域

    如果直接使用全局变量可能会引起冲突,这时通过封装和合理的作用域设置,可以有效避免这些问题: ```javascript var myModule = (function() { var privateVar = "I am private"; function privateMethod() { //...

    浅析matlab多变量拟合.doc

    浅析Matlab多变量拟合 Matlab是一款功能强大的数学软件,可以用于多变量拟合。在本文中,我们将浅析Matlab多变量拟合的原理和实现方法。 一、Matlab多变量拟合的原理 Matlab多变量拟合的原理是基于线性回归分析和...

    浅析JavaScript作用域链、执行上下文与闭包

    JavaScript中的作用域、执行上下文和闭包是理解JavaScript运行机制的关键概念。...通过深入学习和实践,开发者可以更好地掌握JavaScript的精髓,避免因作用域和闭包带来的常见问题,如变量泄露和内存管理不当等。

    浅析php变量修饰符static的使用

    静态变量的特性是在其定义的作用域内仅初始化一次,并且它的值在函数调用之间保持不变。本篇分析将详细探讨static修饰符的使用方法、特点以及相关的注意事项。 首先,静态变量常用于函数内部,目的是保存函数调用...

    浅析Windows Server 2003对域的安全性保障

    《深入探讨Windows Server 2003在域安全性上的保障》 网络安全,特别是域环境的安全,对于企业、教育机构及政府组织来说至关重要。Windows Server 2003作为一款广泛使用的网络操作系统,其在保障域安全性方面具有...

    浅析JavaScript中的变量复制、参数传递和作用域链

    作用域链的查找方式可能导致性能问题,特别是当大量全局变量被使用时。由于每次查找变量都需要遍历作用域链,过多的全局变量会增加查找时间,因此推荐将变量尽可能地限制在局部作用域内。 理解这些基本概念对于...

    5G双域专网解决方案浅析.docx

    "5G双域专网解决方案浅析" 本文旨在解决教育、文旅等行业客户同时访问内网及互联网的需求,提出 5G 双域专网解决方案。通过对双域专网的业务场景进行阐述,并归纳了三类共性需求,然后围绕业务开展的四个要素进行...

    深入浅析javascript中的作用域(推荐)

    作用域定义了变量和函数的可访问范围,它决定了代码块内部以及外部对变量的访问权限。在js中,除了全局作用域,通常只有函数可以创建新的作用域,没有所谓的块级作用域,这点与Java、C等编程语言不同。本文将通过一...

    浅析svm1多分类问题-svm之3 松弛变量(1).rar

    "浅析svm1多分类问题-svm之3 松弛变量(1).rar"这个资料包,主要探讨了SVM在解决多分类问题时引入的松弛变量概念,旨在以简单易懂的方式帮助初学者理解这一技术。 首先,我们要理解SVM的基本原理。SVM通过构建最大...

    浅析JavaScript声明变量

    在块级作用域中使用let声明变量,可以避免var带来的函数作用域问题,以及避免隐式全局变量的错误,这被认为是一种更好的变量声明实践。 在代码组织和变量作用域方面,现代JavaScript编码规范推荐将var声明移动到...

Global site tag (gtag.js) - Google Analytics