PHP垃圾回收机制简单说明
PHP 简单 说明 机制 垃圾 回收
2023-06-13 09:14:23 时间
虽然自己也是PHP的学习者,但之前还真没怎么了解PHP内部的垃圾回收流程,只是在我们的代码中用了unset,null,mysql_close,__destruct等等一些函数去释放对象防止内存溢出而已,所以上网GG下,找到了以下一些说明,作下记录“PHP可以自动进行内存管理,清除不再需要的对象。PHP使用了引用计数(referencecounting)这种单纯的垃圾回收(garbagecollection)机制。每个对象都内含一个引用计数器,每个reference连接到对象,计数器加1。当reference离开生存空间或被设为NULL,计数器减1。当某个对象的引用计数器为零时,PHP知道你将不再需要使用这个对象,释放其所占的内存空间。”
众所周知,PHP引擎本身是用C写的,提到C不能不提的就是GC(垃圾回收).通过PHP手册我们了解到,PHP引擎会自动进行GC动作.那么我们不禁要问,到底它是怎么回收的,&引用操作是不是指针,unset()了一个变量时它是不是真的被回收了呢?这些看似手册有提及的问题,如果仔细分析会发现,远没有那么简单泛泛.也许有人会跳出来说:看PHP源码不就知道了.是的,等你通读了PHP源码后这个问题肯定不在话下了,然本篇要仅从PHP本身来分析这些看似平常却被忽视的小细节,当然了,其中难免水平所限,有所疏漏,热烈欢迎广大phper来共同讨论.
首先咱先看到例子,最简单不过的执行流程了:
Example1:gc.php
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
echo$b."";
?>
不用说%php-fgc.php输出结果非常明了:
hy0kl%php-fgc.php
Iamtest.
好,下一个:
Example2:
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
$b="Iwillchange?";
echo$a."";
echo$b."";
?>
执行结果依然很明显:
hy0kl%php-fgc.php
Iwillchange?
Iwillchange?
君请看:
Example3:
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
unset($a);
echo$a."";
echo$b."";
?>
是不是得想一下下呢?
hy0kl%php-fgc.php
Notice:Undefinedvariable:ain/usr/local/www/apache22/data/test/gc.phponline8
Iamtest.
有点犯迷糊了吗?
君再看:
Example4:
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
unset($b);
echo$a."";
echo$b."";
?>
其实如果Example3理解了,这个与之异曲同工.
hy0kl%php-fgc.php
Iamtest.
Notice:Undefinedvariable:bin/usr/local/www/apache22/data/test/gc.phponline9
君且看:
Example5:
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
$a=null;
echo"$a=".$a."";
echo"$b=".$b."";
?>
猛的第一感觉是什么样的?
hy0kl%php-fgc.php
$a=
$b=
没错,这就是输出结果,对PHPGC已有深入理解的phper不会觉得有什么奇怪,说实话,当我第一次运行这段代码时很意外,却让我对PHPGC有更深刻的理解了.那么下面与之同工的例子自然好理解了.
Example6:
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
$b=null;
echo"$a=".$a."";
echo"$b=".$b."";
?>
OK,如果上面的例子的结果对看官来说无任何细节可言,那您可关闭本窗口了,欢迎有空再来!
下面我们来详细分析GC与引用.
1.所有例子中,创建了一个变量,这个过程通俗一点讲:是在内存中开辟了一块空间,在里面存放了一个字符串Iamtest..PHP内部有个符号表,用来记录各块内存引用计数,那么此时会将这块内存的引用计数加1,并且用一个名为$a的标签(变量)指向这块内存,方便依标签名来操作内存.
2.对变量$a进行&操作,我的理解是找到$a所指向的内存,并为$b建立同样的一引用指向,并将存放字符串Iamtest.的内存块在符号表中引用计数加1.换言之,我们的脚本执行到这一行的时候,存放字符串Iamtest.的那块内存被引用了两次.这里要强调的是,&操作是建立了引用指向,而不是指针,PHP没有指针的概念!同时有人提出说类似于UNIX的文件软链接.可以在一定程度上这么理解:存放字符Iamtest.的那块内存是我们的一个真实的文件,而变量$a与$b是针对真实文件建立的软链接,但它们指向的是同一个真实文件.So,我们看到,在Example2中给$b赋值的同时,$a的值也跟着变化了.与通过某一软链操作了文件类似.
3.在Example3与4中,进行了unset()操作.根据实际的执行结果,可以看出:unset()只是断开这个变量对它原先指向的内存的引用,使变量本身成为没有定义过空引用,所在调用时发出了Notice,并且使那块内存在符号表中引用计数减1,并没有影响到其他指向这块内存的变量.换言之,只有当一块内存在符号表中的引用计数为0时,PHP引擎才会将这块内存回收.
PHP手册
4.0.0unset()becameanexpression.(InPHP3,unset()wouldalwaysreturn1).
这意味着什么?
看看下面的代码与其结果:
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
unset($a);
unset($a);
unset($a);
echo"$a=".$a."";
echo"$b=".$b."";
?>
hy0kl%php-fgc.php
Notice:Undefinedvariable:ain/usr/local/www/apache22/data/test/gc.phponline10
$a=
$b=Iamtest.
第一次unset()的操作已经断开了指向,所以后继的操作不会对符号表的任何内存的引用记数造成影响了.
4.通过Example5&6可以明确无误得出:赋值null操作是相当猛的,它会直接将变量所指向的内存在符号号中的引用计数置0,那这块内存自然被引擎回收了,至于何时被再次利用不得而知,有可能马上被用作存储别的信息,也许再也没有使用过.但是无论如何,原来所有指向那块内存变量都将无法再操作被回收的内存了,任何试图调用它的变量都将返回null.
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
$b=null;
echo"$a=".$a."";
echo"$b=".$b."";
if(null===$a)
{
echo"$aisnull.";
}else
{
echo"Thetypeof$aisunknown.";
}
?>
hy0kl%php-fgc.php
$a=
$b=
$aisnull.
综上所述,充分说明了为什么我们在看开源产品源码的时候,常看到一些比较大的临时变量,或使用完不再调用的重用信息都会被集中或显示的赋值为null了.它相当于UNIX中直接将真实文件干掉了,所有指向它的软链接自然成了空链了.
众所周知,PHP引擎本身是用C写的,提到C不能不提的就是GC(垃圾回收).通过PHP手册我们了解到,PHP引擎会自动进行GC动作.那么我们不禁要问,到底它是怎么回收的,&引用操作是不是指针,unset()了一个变量时它是不是真的被回收了呢?这些看似手册有提及的问题,如果仔细分析会发现,远没有那么简单泛泛.也许有人会跳出来说:看PHP源码不就知道了.是的,等你通读了PHP源码后这个问题肯定不在话下了,然本篇要仅从PHP本身来分析这些看似平常却被忽视的小细节,当然了,其中难免水平所限,有所疏漏,热烈欢迎广大phper来共同讨论.
首先咱先看到例子,最简单不过的执行流程了:
Example1:gc.php
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
echo$b."";
?>
不用说%php-fgc.php输出结果非常明了:
hy0kl%php-fgc.php
Iamtest.
好,下一个:
Example2:
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
$b="Iwillchange?";
echo$a."";
echo$b."";
?>
执行结果依然很明显:
hy0kl%php-fgc.php
Iwillchange?
Iwillchange?
君请看:
Example3:
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
unset($a);
echo$a."";
echo$b."";
?>
是不是得想一下下呢?
hy0kl%php-fgc.php
Notice:Undefinedvariable:ain/usr/local/www/apache22/data/test/gc.phponline8
Iamtest.
有点犯迷糊了吗?
君再看:
Example4:
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
unset($b);
echo$a."";
echo$b."";
?>
其实如果Example3理解了,这个与之异曲同工.
hy0kl%php-fgc.php
Iamtest.
Notice:Undefinedvariable:bin/usr/local/www/apache22/data/test/gc.phponline9
君且看:
Example5:
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
$a=null;
echo"$a=".$a."";
echo"$b=".$b."";
?>
猛的第一感觉是什么样的?
hy0kl%php-fgc.php
$a=
$b=
没错,这就是输出结果,对PHPGC已有深入理解的phper不会觉得有什么奇怪,说实话,当我第一次运行这段代码时很意外,却让我对PHPGC有更深刻的理解了.那么下面与之同工的例子自然好理解了.
Example6:
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
$b=null;
echo"$a=".$a."";
echo"$b=".$b."";
?>
OK,如果上面的例子的结果对看官来说无任何细节可言,那您可关闭本窗口了,欢迎有空再来!
下面我们来详细分析GC与引用.
1.所有例子中,创建了一个变量,这个过程通俗一点讲:是在内存中开辟了一块空间,在里面存放了一个字符串Iamtest..PHP内部有个符号表,用来记录各块内存引用计数,那么此时会将这块内存的引用计数加1,并且用一个名为$a的标签(变量)指向这块内存,方便依标签名来操作内存.
2.对变量$a进行&操作,我的理解是找到$a所指向的内存,并为$b建立同样的一引用指向,并将存放字符串Iamtest.的内存块在符号表中引用计数加1.换言之,我们的脚本执行到这一行的时候,存放字符串Iamtest.的那块内存被引用了两次.这里要强调的是,&操作是建立了引用指向,而不是指针,PHP没有指针的概念!同时有人提出说类似于UNIX的文件软链接.可以在一定程度上这么理解:存放字符Iamtest.的那块内存是我们的一个真实的文件,而变量$a与$b是针对真实文件建立的软链接,但它们指向的是同一个真实文件.So,我们看到,在Example2中给$b赋值的同时,$a的值也跟着变化了.与通过某一软链操作了文件类似.
3.在Example3与4中,进行了unset()操作.根据实际的执行结果,可以看出:unset()只是断开这个变量对它原先指向的内存的引用,使变量本身成为没有定义过空引用,所在调用时发出了Notice,并且使那块内存在符号表中引用计数减1,并没有影响到其他指向这块内存的变量.换言之,只有当一块内存在符号表中的引用计数为0时,PHP引擎才会将这块内存回收.
PHP手册
4.0.0unset()becameanexpression.(InPHP3,unset()wouldalwaysreturn1).
这意味着什么?
看看下面的代码与其结果:
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
unset($a);
unset($a);
unset($a);
echo"$a=".$a."";
echo"$b=".$b."";
?>
hy0kl%php-fgc.php
Notice:Undefinedvariable:ain/usr/local/www/apache22/data/test/gc.phponline10
$a=
$b=Iamtest.
第一次unset()的操作已经断开了指向,所以后继的操作不会对符号表的任何内存的引用记数造成影响了.
4.通过Example5&6可以明确无误得出:赋值null操作是相当猛的,它会直接将变量所指向的内存在符号号中的引用计数置0,那这块内存自然被引擎回收了,至于何时被再次利用不得而知,有可能马上被用作存储别的信息,也许再也没有使用过.但是无论如何,原来所有指向那块内存变量都将无法再操作被回收的内存了,任何试图调用它的变量都将返回null.
<?php
error_reporting(E_ALL);
$a="Iamtest.";
$b=&$a;
$b=null;
echo"$a=".$a."";
echo"$b=".$b."";
if(null===$a)
{
echo"$aisnull.";
}else
{
echo"Thetypeof$aisunknown.";
}
?>
hy0kl%php-fgc.php
$a=
$b=
$aisnull.
综上所述,充分说明了为什么我们在看开源产品源码的时候,常看到一些比较大的临时变量,或使用完不再调用的重用信息都会被集中或显示的赋值为null了.它相当于UNIX中直接将真实文件干掉了,所有指向它的软链接自然成了空链了.
相关文章
- php和asp网络验证码,Verifycode 1个简单的网页图片验证码的示例程序,基本上现有 字和字母都可以识别。 WEB(ASP,PHP,…) 238万源代码下载- www.pudn.com…
- 新手php环境一键安装包,PHP运行环境一键安装包(phpStudy2013)
- windows宝塔PHP出现500怎么处理?
- [PHP] apache在worker模式配置fastcgi使用php-fpm详解编程语言
- 命令在PHP中执行Linux命令的方法(php执行linux)
- PHP搭建Redis环境指南(php配置redis)
- 简单步骤实现PHP连接Redis集群(php连接redis集群)
- 实现【使用Redis实现PHP抢红包迅速拿奖励】(php抢红包redis)
- 加载PHP MySQL 遍历加载 快速、简单的数据查询方法!(phpmysql遍历)
- 实现PHP程序Redis连接池的简单方法(redis连接池php)
- 使用PHP操作Redis:简单灵活的方法(php如何使用redis)
- 利用PHP操作Redis变得更简单(php中redis的使用)
- 使用PHP操作Redis实例:简单高效的数据存储方案(php操作redis实例)
- PHP中使用Redis简单取值(redis取值php)
- PHP链接MSSQL:解开空白之谜(php链接mssql空白)
- PHP程序实现MSSQL数据库的连接(php程序连接mssql)
- 使用Redis需要PHP版本满足特定要求(redis需要php版本)
- 使用PHP连接Redis应用的指南(redis连接 php)
- 给php新手谈谈我的学习心得
- 在PHP中使用Sockets从Usenet中获取文件
- php生成随机数或者字符串的代码
- 简单的php验证图片生成函数
- PHP中开发XML应用程序之基础篇添加节点删除节点查询节点查询节
- PHP和Mysqlweb应用开发核心技术第1部分Php基础-1开始了解php
- PHP中“简单工厂模式”实例代码讲解
- php实现压缩多个CSS与JS文件的方法
- PHP编译安装中遇到的两个错误和解决方法