PHP变量作用域和全局变量

2018-07-06  本文已影响12人  半亩房顶

转自:https://blog.csdn.net/u010412301/article/details/53958521

PHP作为一个脚本的解释型语言,弱变量的特点和执行完释放资源的特点诸城,PHP7的强势加入更是在后端语言的群雄中掀起了一阵强烈的旋风。好了,由于本人我平时也不怎么注意变量的作用域,由此写这篇文章也算是自我提醒。

而PHP的语法特点和c++也比较像,再加上_POST,FILE等全局变量和__construct()和__destruct()等魔术变量使得开发更显得方便许多。

但有的人很不习惯PHP中的变量作用域,PHP中函数变量和全局是完全隔绝的,也就是无法相互访问。

<?php
    $test = 'hello,world';
     abc(); //这里什么都不输出,因为访问不到$test变量
    function abc(){
            echo($test);
    }
?>
image

global和$GLOBALS[]

我们可以使用global关键字来声明变量,上面的例子就变成了这样

$test = 'hello,world';
     abc(); 
    function abc(){
        global $test;
            echo $test;
    }

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

$test = 'hello,world';
    function abc(){
        echo $GLOBALS['test'];
    }
    abc();

原来以为global和$GLOBALS除了写法不一样以为,其他都一样,可是在实际应用中发现,2者的区别还是很大的! 看这个例子:

function test1() { 
    global $v1, $v2; 
    $v2 =& $v1; 
} 
function test2() { 
    $GLOBALS['v3'] =& $GLOBALS['v1']; 
} 
$v1 = 1; 
$v2 = $v3 = 0; 
test1(); 
echo $v2 ."\n"; 
test2(); 
echo $v3 ."\n";

为什么是这样:

image

不应该是两个1吗?我们再看一个例子

function test() { 
    global $a; 
    unset($a); 
} 
$a = 1; 
test(); 
echo $a;
image

明明是unset了呀,为什么还会打印出来呢?
众所周知,我们的function里面的永远是个私有变量,unset的确是起作用了,它unset了一个global 的值呀,而global在函数产生一个指向函数外部变量的别名变量,而不是真正的函数外部变量;$GLOBALS[]确确实实调用是外部的变量,函数内外会始终保持一致!

use()

大家对use()的理解是不是还是命名空间的使用,PHP 命名空间支持有两种使用别名或导入方式:为类名称使用别名,或为命名空间名称使用别名, 别名通过操作符 use 来实现。
但我们今天说的是这种形式:function use(){}
php5.3新增闭包语法

//普通
$a="hello,world!";
$test = function () use($a){
    echo $a;
};
$test();
//引用对象
$ob=(object)array('name' => 'gbw');
$test2 = function () use($ob){
    var_dump($ob);
};
$test2();

PHP闭包的特性并没有太大惊喜,其实用CLASS就可以实现类似甚至强大得多的功能,更不能和js的闭包相提并论。所以这种写法也并不是很常见。

备注:

use() 有坑注意

实际使用中,我尝试了 use() 的用法,但是发现报错,故查了下资料,结论如下:
PHP 7 可以使用一个 use 从同一个 namespace 中导入类、函数和常量:
实例
// PHP 7 之前版本需要使用多次 use
use some\namespace\ClassA;
use some\namespace\ClassB;
use some\namespace\ClassC as C;

use function some\namespace\fn_a;
use function some\namespace\fn_b;
use function some\namespace\fn_c;

use const some\namespace\ConstA;
use const some\namespace\ConstB;
use const some\namespace\ConstC;

// PHP 7+ 之后版本可以使用一个 use 导入同一个 namespace 的类
use some\namespace{ClassA, ClassB, ClassC as C};
use function some\namespace{fn_a, fn_b, fn_c};
use const some\namespace{ConstA, ConstB, ConstC};

上一篇下一篇

猜你喜欢

热点阅读