从php代码分析php的GC(垃圾回收)机制

首先咱先看到例子,最简单不过的执行流程了:
Example 1: gc.php
<?php 
error_reporting
(E_ALL); 
$a 'I am test.'
$b = & $a

echo 
$b ."\n"; 

?>

不用说 % php -f gc.php 输出结果很是明了:
hy0kl% php -f gc.php 
I am test.


好,下一个:
Example 2:
<?php 
error_reporting
(E_ALL); 
$a 'I am test.'
$b = & $a

$b 'I will change?';                                                           

echo 
$a ."\n"
echo 
$b ."\n"; 

?>
执行结果依然很明显:
hy0kl% php -f gc.php 
I will change?
I will change?


君请看:
Example 3:
<?php 
error_reporting
(E_ALL); 
$a 'I am test.'
$b = & $a;  

unset(
$a); 

echo 
$a ."\n"
echo 
$b ."\n";
?>
是否是得想一下下呢?
hy0kl% php -f gc.php 
Notice: Undefined variable: a in /usr/local/www/apache22/data/test/gc.php on line 8
I am test.

有点犯迷糊了吗?

君再看:
Example 4:
<?php 
error_reporting
(E_ALL); 
$a 'I am test.'
$b = & $a

unset(
$b);                                                                       

echo 
$a ."\n"
echo 
$b ."\n";

?>
其实若是 Example 3 理解了,这个与之殊途同归.
hy0kl% php -f gc.php 
I am test.
Notice: Undefined variable: b in /usr/local/www/apache22/data/test/gc.php on line 9


君且看:
Example 5:
<?php 
error_reporting
(E_ALL); 
$a 'I am test.'
$b = & $a

$a null

echo 
'$a = '$a ."\n"
echo 
'$b = '$b ."\n"; 

?>
猛的第一感受是什么样的?
hy0kl% php -f gc.php 
$a = 
$b =

没错,这就是输出结果,对 PHP GC 已有深刻理解的 phper 不会以为有什么奇怪,说实话,当我第一次运行这段代码时很意外,却让我对 PHP GC 有更深入的理解了.那么下面与之同工的例子天然好理解了.

Example 6:
<?php                                                                            
error_reporting
(E_ALL); 
$a 'I am test.'
$b = & $a

$b null

echo 
'$a = '$a ."\n"
echo 
'$b = '$b ."\n"; 

?>

OK,若是上面的例子的结果对看官来讲无任何细节可言,那您可关闭本窗口了,欢迎有空再来!

下面咱们来详细分析 GC 与引用.
1. 全部例子中,建立了一个变量,这个过程通俗一点讲:是在内存中开辟了一块空间,在里面存放了一个字符串 I am test. . PHP 内部有个符号表,用来记录各块内存引用计数,那么此时会将这块内存的引用计数 加 1,而且用一个名为 $a 的标签(变量)指向这块内存,方便依标签名来操做内存.

2. 对变量 $a 进行 & 操做,个人理解是找到 $a 所指向的内存,并为 $b 创建一样的一引用指向,并将存放字符串 I am test. 的内存块在符号表中引用计数 加 1.换言之,咱们的脚本执行到这一行的时候,存放字符串 I am test. 的那块内存被引用了两次.这里要强调的是,& 操做是创建了引用指向,而不是指针, PHP 没有指针的概念!同时有人提出说相似于 UNIX 的文件软连接.能够在必定程度上这么理解: 存放字符 I am test. 的那块内存是咱们的一个真实的文件,而变量 $a 与 $b 是针对真实文件创建的软连接,但它们指向的是同一个真实文件. So, 咱们看到,在 Example 2  中给 $b 赋值的同时, $a 的值也跟着变化了.与经过某一软链操做了文件相似.

3. 在 Example 3 与 4 中,进行了 unset() 操做.根据实际的执行结果,能够看出: unset() 只是断开这个变量对它原先指向的内存的引用,使变量自己成为没有定义过空引用,所在调用时发出了 Notice ,而且使那块内存在符号表中引用计数 减 1,并无影响到其余指向这块内存的变量.换言之,只有当一块内存在符号表中的引用计数为 0 时, PHP 引擎才会将这块内存回收.
PHP 手册
4.0.0                 unset() became an expression. (In PHP 3,         unset() would always return 1). 
这意味着什么?
看看下面的代码与其结果:
<?php 
error_reporting
(E_ALL); 
$a 'I am test.'
$b = & $a

unset(
$a); 
unset(
$a); 
unset(
$a); 

echo 
'$a = '$a ."\n"
echo 
'$b = '$b ."\n"; 

?>
hy0kl% php -f gc.php 

Notice: Undefined variable: a in /usr/local/www/apache22/data/test/gc.php on line 10
$a = 
$b = I am test.
第一次 unset() 的操做已经断开了指向,因此后继的操做不会对符号表的任何内存的引用记数形成影响了.

4. 经过 Example 5 & 6 能够明确无误得出: 赋值 null操做是至关猛的,它会直接将变量所指向的内存在符号号中的引用计数置 0,那这块内存天然被引擎回收了,至于什么时候被再次利用不得而知,有可能立刻被用做存储别的信息,也许再也没有使用过.可是不管如何,原来全部指向那块内存变量都将没法再操做被回收的内存了,任何试图调用它的变量都将返回 null.

<?php 
error_reporting
(E_ALL); 
$a 'I am test.'
$b = & $a

$b null

echo 
'$a = '$a ."\n"
echo 
'$b = '$b ."\n"

if (
null === $a
{                                                                                
echo 
'$a is null.';     
} else 

echo 
'The type of $a is unknown.';     
}
 

?>
hy0kl% php -f gc.php 
$a = 
$b = 
$a is null.


综上所述,充分说明了为何咱们在看开源产品源码的时候,常看到一些比较大的临时变量,或使用完再也不调用的重用信息都会被集中或显示的赋值为 null 了.它至关于 UNIX 中直接将真实文件干掉了,全部指向它的软连接天然成了空链了.
以前在讨论到这些细节点时有不少想固然的念头,在实际的执行了测试代码后才发现: 哦,原来如此!php

相关文章
相关标签/搜索