转自: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);
}
?>
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";
为什么是这样:
不应该是两个1吗?我们再看一个例子
function test() {
global $a;
unset($a);
}
$a = 1;
test();
echo $a;
明明是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};