众所周知, PHP 引擎本身是用 C 写的,提到 C 不能不提的就是 GC(垃圾回收).通过 PHP 手册我们了解到, PHP 引擎会自动进行 GC 动作.那么我们不禁要问,到底它是怎么回收的, & 引用操作是不是指针, unset()了一个变量时它是不是真的被回收了呢?这些看似手册有提及的问题,如果仔细分析会发现,远没有那么简单泛泛.也许有人会跳出来说:看 PHP源码不就知道了.是的,等你通读了 PHP 源码后这个问题肯定不在话下了,然本篇要仅从 PHP本身来分析这些看似平常却被忽视的小细节,当然了,其中难免水平所限,有所疏漏,热烈欢迎广大 phper 来共同讨论.
首先咱先看到例子,最简单不过的执行流程了:
Example 1: gc.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:
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:
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:
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:
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:
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).
这意味着什么?
看看下面的代码与其结果:
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.
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 5.3引入了Zend GC(Garbage Collector),这是一种基于标记-清除算法的垃圾回收机制。 在Zend GC中,PHP会在特定时刻进行一次完整的遍历,标记所有可达(仍然在使用)的变量。然后,所有未被...
为了解决这个问题,从PHP 5.3开始引入了一种更为复杂的垃圾回收机制,称为"标记-清除"(Mark-and-Sweep)。在这个机制中,PHP会定期启动一个垃圾回收周期,遍历所有活动的变量,标记那些可以从全局符号表可达的变量...
从PHP 5.3开始,引入了更为复杂的垃圾回收机制,通过周期性地检查并清除不再可达的对象来解决这个问题。 在PHP中,通过`is_ref__gc`字段可以区分变量是普通变量还是引用。如果`is_ref__gc`为1,则表示该变量是一个...
PHP strtotime应用经验之谈PHP memory_get_usage()管理内存PHP unset全局变量运用问题详解PHP unset()函数销毁变量教你快速实现PHP全站权限验证一、PHP 垃圾回收机制(Garbage Collector 简称GC) 在PHP中,没有任何...
总结来说,PHP的引用计数器和垃圾回收机制共同确保了内存的有效管理,允许程序员编写出高效且内存安全的代码。理解这些机制有助于优化PHP程序,避免潜在的内存泄漏问题,并充分利用PHP的资源管理功能。
PHP7中的垃圾回收机制针对的是PHP动态类型系统中的复杂数据类型,比如字符串、数组和对象,这些数据结构中都包含了一个专门用于垃圾回收的gc(Garbage Collection)头部。 PHP7中的垃圾回收主要依靠引用计数...
PHP的垃圾回收(GC)机制是为了管理内存而设计的,主要目标是自动识别并清理不再使用的对象,防止内存泄漏。在PHP中,垃圾回收主要依赖于引用计数(reference counting)这一策略。每创建一个对象,它都会有一个内置...
本文实例讲述了PHP session垃圾回收机制。分享给大家供大家参考,具体如下: session过期时间 在php.ini文件中有这样一个配置,这个配置表示session文件过期时间,默认的话是1440秒,也就是24分钟,这个24分钟是...
本篇文章将深入探讨“HP-socket 扩展 Session 包含GC垃圾回收机制”这一易语言实现的Session管理方案。 首先,我们要理解“HP-socket扩展”。HP(Happy Programming)可能是开发者自定义的一个框架或库,这里的...
在PHP编程语言中,析构函数`__destruct()`和垃圾回收机制(GC)是两个重要的概念,它们在管理内存和确保程序高效运行方面起到关键作用。这篇文章将详细讲解这两个概念。 首先,我们来理解析构函数`__destruct()`. ...
在深入探讨PHP垃圾回收机制对内存泄露的处理之前,首先需要了解PHP中变量的存储...然而,垃圾回收机制的引入也带来了一定的性能开销,因此开发者在编写PHP代码时仍然需要谨慎地处理变量引用,以避免不必要的性能损失。
在深入探讨PHP的垃圾回收机制之前,首先要明白的是,垃圾回收(Garbage Collection, GC)是编程语言中用于自动管理内存的一种技术,它的目的是自动识别并释放不再使用的内存空间,以避免内存泄漏。PHP作为一门动态...
一、为什么需要GC 应用程序对资源操作,通常简单分为以下几个步骤: 1、为对应的资源分配内存 2、初始化内存 3、使用资源 4、清理资源 5、释放内存 应用程序对资源(内存使用)管理的方式,常见的一般有如下几种: 1...