Maison  >  Article  >  php教程  >  PHP Global与$GLOBALS变量作用域与区别

PHP Global与$GLOBALS变量作用域与区别

WBOY
WBOYoriginal
2016-05-25 16:48:291298parcourir

在php中变量有很多种如普通的变量及全局变量(Global与$GLOBALS),本文章来给大家介绍在php中Global与$GLOBALS的用法区别.

Global,全局变量

PHP Global变量在实际应用中会发现许多问题需要我们不断的去完善处理.我们在这篇文章中就针对PHP Global变量出现的问题给出了一些具体的解决办法,PHP hack的使用技巧详解,代码实现PHP GTK写文本查看器,网站开发中PHP语言优缺点.

如何正确实现PHP function函数扩展

PHP error_log()函数处理错误日志

1:PHP Global变量的作用是定义全局变量,但是这个全局变量不是应用于整个网站,而是应用于当前页面,包括include或require的所有文件

实例代码如下:

$a=123;
function aa()  
{  
    Global $a;   
    //如果不把$a定义为global变量,函数体内是不能访问$a的  
    echo $a;  
}  
aa();

总结:

在函数体内定义的PHP Global变量,函数体外可以使用,在函数体外定义的global变量不能在函数体内使用,

实例代码如下:

$glpbal $a;   
$a=123;    
function f()   
{   
    echo $a; //错误,   
}

再看看下面一例,实例代码如下:

function f()   
{   
    global $a;  $a=123;   
}    
f();   
echo $a;
//正确,可以使用

2:PHP Global变量问题解析:

question:我在config.inc.php中定义了一些变量($a),在别的文件中函数外部 include("config.inc.php"),函数内部需要使用这些变量$a,如果没有声明的话,echo $a是打印不出来任何东西的.因此声明global $a,但是有很多函数和很多变量.总不能不断重复的这样声明吧?有什么好的解决办法,请指点.

answer1:先在config.inc.php里定义常量:define(常量名,常量值),再在其他需要用到的地方require 'config.inc.php',然后就能在这个文件里直接使用这个常量了.

answer2:我也有个办法,就是定义数组,如$x[a],$x,那样就只要声明global $x一个了.

answer3:我试了你的这个方法,不行啊.

answer4:改你的php.ini文件.

设置PHP Global变量 为 on,下面我们看看复杂点的:

实例代码如下:

//A.php 文件
<?php
function Test_Global() {
    include &#39;B.php&#39;;
    Test();
}
$a = 0;
Test_Global();
echo $a;
?>
//B.php 文件 
<?php
function Test() {
    global $a; //申明函数体Sum内使用的$a变量为global全局变量
    $a = 1;
}
?>

为什么输出的却是0?!!

在用户自定义函数中,一个局部函数范围将被引入.任何用于函数内部的变量按缺省情况将被限制在局部函数范围内(包括include 和 require 导入的文件内的变量)!

解释:A.php文件的内Test_Global是定义好的第三方函数,该函数用include导入了B.php文件内的$a的global全局变量,所以$a被限制在Test_Global局部函数范围内,所以B.php文件内的$a的作用范围都在Test_Global内,而不是作用了整个A.php内….

解决方案:

1. 冲出局部函数实例代码如下:

//A.php 文件 
<?php
function Test_Global() {
    Test();
}
include &#39;B.php&#39;; //将include 从局部Test_Global函数中移出
$a = 0;
Test_Global();
echo $a;
?>
//B.php 文件 
<?php
function Test() {
    global $a;
    $a = 1;
}
?>

global和$GLOBALS的区别

php中global和$GLOBALS不仅仅是写法不一样以为,2者的区别还是很大的,在实际应用中需要注意!

先看下面的实例代码:

<?php
// 例子1
function test_global() {
    global $var1, $var2;
    $var2 = & $var1;
}
function test_globals() {
    $GLOBALS[&#39;var3&#39;] = & $GLOBALS[&#39;var1&#39;];
}
$var1 = 5;
$var2 = $var3 = 0;
test_global();
print $var2 . "\n";
test_globals();
print $var3 . "\n";
?>

执行结果为:0 5

怎么会这样呢?不应该是2个5吗?怎么会出现1个0和1个5呢?

恩,我们保留以上问题,深入分析$GLOBALS和global的原理!我们都知道变量其实是相应物理内存在代码中的"代号"而已引用php手册的$GLOBALS的解释:

Global 变量:$GLOBALS,注意: $GLOBALS 在 PHP 3.0.0 及以后版本中适用.由所有已定义全局变量组成的数组.变量名就是该数组的索引.这是一个"superglobal",或者可以描述为自动全局变量.也就是说上面代码中的$var1和$GLOBALS['var1']是指的同一变量,而不是2个不同的变量!下面来分析global到底做了什么?

引用php手册的global的解释:

如果在一个函数内部给一个声明为 global 的变量赋于一个引用,该引用只在函数内部可见.可以通过使用 $GLOBALS 数组避免这一点.我们都知道php中的函数所产生的变量都是函数的私有变量,那么global关键字产生的变量也肯定逃不出这个规则,为什么这么说呢,

请看下面的代码:

<?php
// 例子2
function test() {
    global $a;
    unset($a);
}
$a = 1;
test();
print $a;
?>

执行结果为:1

为什么会输出1呢?不是已经把$a给unset了吗?unset失灵了?php的bug?

都不是,其实unset起作用了,是把test函数中的$a给unset掉了,可以在函数test()中加入print $a;来测试!

接着回到上面的例子1,看test_global中的这一代码"$var2 = &$var1;",上面是一个引用赋值运算,也就是$var2将指向var1所指向的物理内存地址,所以例子1执行过test_global函数以后,变量的变化只在函数的局部产生效应,在函数外部$var2的指向物理内存地址并没有变化,还是它自己.此时,就能理解为什么例子1执行完以后,$var2是0,而$var3是5了!

所以我们得出一个结论,在函数中global和$GLOBALS[]的区别在于:

global在函数产生一个指向函数外部变量的别名变量,而不是真正的函数外部变量,一但改变了别名变量的指向地址,就会发生一些意料不到情况,例如例子1.

$GLOBALS[]确确实实调用是外部的变量,函数内外会始终保持一致,可以对照下面两个列子再加深下印象:

global:

实例代码如下:

<?php
function myfunction() {
    global $bar;
    unset($bar);
}
$bar = "someting";
myfunction();
echo $bar;
?>

输出:someting

$GLOBALS[]: 

<?php
function foo() {
    unset($GLOBALS[&#39;bar&#39;]);
}
$bar = "something";
foo();
echo $bar;
?>

输出:空

当按照上面的思路理解后,碰到下面的情况是不是又有些晕呢?

实例代码如下:

<?php
$a = 1;
$b = 2;
function Sum() {
    global $a, $b;
    $b = $a + $b;
}
Sum();
echo $b;
?>

输出将是 "3".在函数中申明了全局变量 $a 和 $b,任何变量的所有引用变量都会指向到全局变量.

怎么不是2呢,在函数外部不是不影响吗,请注意$b在函数中并没有通过引用修改,而是修改的$b指向物理内存的值,因此外部输入为3.

php中global和$GLOBALS不仅仅是写法不一样以为,2者的区别还是很大的,在实际应用中需要注意!

先看下面的PHP代码例子:

<?php
// 例子1
function test_global() {
    global $var1, $var2;
    $var2 = & $var1;
}
function test_globals() {
    $GLOBALS[&#39;var3&#39;] = & $GLOBALS[&#39;var1&#39;];
}
$var1 = 5;
$var2 = $var3 = 0;
test_global();
print $var2 . "\n";
test_globals();
print $var3 . "\n";
?>

执行结果为:0 5

怎么会这样呢?不应该是2个5吗?怎么会出现1个0和1个5呢?

恩,我们保留以上问题,深入分析$GLOBALS和global的原理!我们都知道变量其实是相应物理内存在代码中的"代号"而已,引用php手册的$GLOBALS的解释:

Global 变量:$GLOBALS,注意: $GLOBALS 在 PHP 3.0.0 及以后版本中适用.

由所有已定义全局变量组成的数组.变量名就是该数组的索引.这是一个"superglobal",或者可以描述为自动全局变量.也就是说上面代码中的$var1和$GLOBALS['var1']是指的同一变量,而不是2个不同的变量!

下面来分析global到底做了什么?

引用php手册的global的解释:

如果在一个函数内部给一个声明为 global 的变量赋于一个引用,该引用只在函数内部可见.可以通过使用 $GLOBALS 数组避免这一点.我们都知道php中的函数所产生的变量都是函数的私有变量,那么global关键字产生的变量也肯定逃不出这个规则,为什么这么说呢,看下面的代码:

实例代码如下:

<?php
// 例子2
function test() {
    global $a;
    unset($a);
}
$a = 1;
test();
print $a;
?>

执行结果为:1

为什么会输出1呢?不是已经把$a给unset了吗?unset失灵了?php的bug?

都不是,其实unset起作用了,是把test函数中的$a给unset掉了,可以在函数test()中加入print $a;来测试!

接着回到上面的例子1,看test_global中的这一代码"$var2 =& $var1;",上面是一个引用赋值运算,也就是$var2将指向var1所指向的物理内存地址,所以例子1执行过test_global函数以后,变量的变化只在函数的局部产生效应,在函数外部$var2的指向物理内存地址并没有变化,还是它自己.

此时,就能理解为什么例子1执行完以后,$var2是0,而$var3是5了!

所以我们得出一个结论,在函数中global和$GLOBALS[]的区别在于:

global在函数产生一个指向函数外部变量的别名变量,而不是真正的函数外部变量,一但改变了别名变量的指向地址,就会发生一些意料不到情况,例如例子1.

$GLOBALS[]确确实实调用是外部的变量,函数内外会始终保持一致,可以对照下面两个列子再加深下印象:

global:

实例代码如下:

<?php
function myfunction() {
    global $bar;
    unset($bar);
}
$bar = "someting";
myfunction();
echo $bar;
?>

输出:someting

实例代码如下:

$GLOBALS[]: 

<?php
function foo() {
    unset($GLOBALS[&#39;bar&#39;]);
}
$bar = "something";
foo();
echo $bar;
?>

输出:空

当按照上面的思路理解后,碰到下面的情况是不是又有些晕呢?

实例代码如下:

<?php
$a = 1;
$b = 2;
function Sum() {
    global $a, $b;
    $b = $a + $b;
}
Sum();
echo $b;
?>

输出将是 "3".在函数中申明了全局变量 $a 和 $b,任何变量的所有引用变量都会指向到全局变量.

怎么不是2呢,在函数外部不是不影响吗,请注意$b在函数中并没有通过引用修改,而是修改的$b指向物理内存的值,因此外部输入为3


Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn