服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|JAVA教程|ASP教程|

服务器之家 - 编程语言 - PHP教程 - PHP垃圾回收机制简单说明

PHP垃圾回收机制简单说明

2019-11-04 13:52php教程网 PHP教程

最近有看到一份面试题目,内容大概为阐述PHP的垃圾回收机制是怎么一回事。

虽然自己也是PHP的学习者,但之前还真没怎么了解PHP内部的垃圾回收流程,只是在我们的代码中用了unset,null,mysql_close,__destruct等等一些函数去释放对象防止内存溢出而已,所以上网GG下,找到了以下一些说明,作下记录“PHP可以自动进行内存管理,清除不再需要的对象。PHP使用了引用计数(reference counting)这种单纯的垃圾回收(garbage collection)机制。每个对象都内含一个引用计数器,每个reference连接到对象,计数器加1。当reference离开生存空间或被设为NULL,计数器减1。当某个对象的引用计数器为零时,PHP知道你将不再需要使用这个对象,释放其所占的内存空间。”

众所周知, PHP 引擎本身是用 C 写的,提到 C 不能不提的就是 GC(垃圾回收).通过 PHP 手册 我们了解到, PHP 引擎会自动进行 GC 动作.那么我们不禁要问,到底它是怎么回收的, & 引用操作是不是指针, unset() 了一个变量时它是不是真的被回收了呢?这些看似手册有提及的问题,如果仔细分析会发现,远没有那么简单泛泛.也许有人会跳出来说:看 PHP 源码不就知道了.是的,等你通读了 PHP 源码后这个问题肯定不在话下了,然本篇要仅从 PHP 本身来分析这些看似平常却被忽视的小细节,当然了,其中难免水平所限,有所疏漏,热烈欢迎广大 phper 来共同讨论. 

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

echo $b ." "; 
?> 

不用说 % 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 ." "; 
echo $b ." "; 
?> 
执行结果依然很明显: 
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 ." "; 
echo $b ." "; 
?> 
是不是得想一下下呢? 
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 ." "; 
echo $b ." "; 
?> 
其实如果 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 ." "; 
echo '$b = '. $b ." "; 
?> 
猛的第一感觉是什么样的? 
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 ." "; 
echo '$b = '. $b ." "; 
?> 

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 ." "; 
echo '$b = '. $b ." "; 
?> 
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 ." "; 
echo '$b = '. $b ." "; 

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教程PHP-FPM的配置与优化讲解

    PHP-FPM的配置与优化讲解

    今天小编就为大家分享一篇关于PHP-FPM的配置与优化讲解,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看...

    CODETC1962019-05-30
  • PHP教程php curl优化下载微信头像的方法总结

    php curl优化下载微信头像的方法总结

    这篇文章主要给大家介绍了关于php curl优化下载微信头像的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值...

    soledad4222019-09-07
  • PHP教程php 可变函数使用小结

    php 可变函数使用小结

    PHP 支持可变函数的概念。这意味着如果一个变量名后有圆括号,PHP 将寻找与变量的值同名的函数,并且尝试执行它。本文重点给大家介绍php 可变函数使用...

    烟熏妆4392019-09-27
  • PHP教程PHP实现上传图片到数据库并显示输出的方法

    PHP实现上传图片到数据库并显示输出的方法

    这篇文章主要介绍了PHP实现上传图片到数据库并显示输出的方法,结合实例形式分析了php采用二进制形式存储图片及读取显示的相关操作技巧,需要的朋友可...

    m_nanle_xiaob2492019-09-29
  • PHP教程PHP实现随机数字、字母的验证码功能

    PHP实现随机数字、字母的验证码功能

    本文通过实例代码给大家介绍了PHP实现随机数字、字母的验证码功能,代码简单易懂,非常不错,具有一定得参考借鉴价值,需要得朋友参考下吧 ...

    我们家的小常客1712019-09-17
  • PHP教程Mac系统下安装PHP Xdebug

    Mac系统下安装PHP Xdebug

    本文给大家详细讲述了在Mac系统下安装PHP Xdebug的详细步骤和流程,一起学习下吧。 ...

    laozhang4942019-10-14
  • PHP教程PHP zip压缩包操作类完整实例

    PHP zip压缩包操作类完整实例

    这篇文章主要介绍了PHP zip压缩包操作类,结合完整实例形式分析了php封装的针对zip文件解压、压缩、递归创建、读取等相关操作,需要的朋友可以参考下 ...

    小虎的码农博客1972019-09-23
  • PHP教程dirname(__FILE__)的含义和应用说明

    dirname(__FILE__)的含义和应用说明

    这篇文章主要介绍了dirname(__FILE__)的含义和应用说明,需要的朋友可以参考下 ...

    mdxy-dxy1652019-10-15