$GLOBALS与global区别 & 变量销毁机制

代码一:


$var1 = 1;
function test1(){
    global $var1;  #等价于 $var1 = &$GLOBALS['var1']; 这里的$var1跟外面的$var1是不同的指针,但指向同样的数据
    unset($var1);   #当你unset一个引用,只是断开了变量名和变量内容之间的绑定,这并不意味着变量内容被销毁了.
    echo $var1;     #PHP Notice:  Undefined variable: var1
}
test1();
echo $var1;    #此处输出1

代码二:


$var1 = 1;
function test1(){
    global $var1;
    unset($GLOBALS['var1']);
    echo $var1;  #输出1
}

test1();
echo $var1;  #PHP Notice:  Undefined variable: var1

结论:

1$GLOBALS['var2'] 和外部的 $var2 是同一个指针,指向存储值为2的内存地址
2global $var1 是外部的 $val1 的指针的一个复制指针,也指向存储值为1的内存地址(同引用符号&)
3、无论$GLOBALS['var1']或者global $var1,因为指向的值内存地址一样,所以都可以修改外部变量的值。

所以:
unset($GLOBALS['var1']) 操作同时销毁了外部的 $var1
test1 函数下进行 global $val1; unset($val1) 不会销毁外部的$var1

php 变量管理符号表(php以符号表的形式管理变量):

php符号表理解:运行中所有变量名称都记录在里面,php来维护,具体的数据存储在内存中,php就是根据这个符号表去回收没有用到的变量空间的,释放内存空间

$GLOBALS与global区别 & 变量销毁机制_第1张图片

当你unset一个变量,只是断开了变量名和变量内容之间的绑定,这并不意味着变量内容被销毁了.此时如上图所示,unset时符号表对应的refcount - 1  当 refcount = 0 时,php才会启动垃圾回收,真正释放内存
php中引用采用的是“写时拷贝”的原理,就是除非发生写操作,指向同一个地址的变量或者对象是不会被拷贝的。

通俗的讲 

1:如果有下面的代码 $a="ABC"; $b=$a; 其实此时 $a$b都是指向同一内存地址 而并不是$a$b占用不同的内存

2:如果在上面的代码基础上再加上如下代码 $a="EFG";  由于$a$b所指向的内存的数据要重新写一次了,此时Zend核心会自动判断 自动为$b生产一个$a的数据拷贝,重新申请一块内存进行存储

$GLOBALS与global区别 & 变量销毁机制_第2张图片

你可能感兴趣的:(PHP)