从几个简单的程序看PHP的垃圾回收机制

每一种计算机语言都有自己的自动垃圾回收机制,让程序员不必过分关心程序内存分配,php也不例外,但是在面向对象编程(OOP)编程中,有些对象需要显式的销毁,防止程序执行内存溢出。

一、PHP 垃圾回收机制(Garbage Collector 简称GC)永盈会娱乐城

在PHP中,没有任何变量指向这个对象时,这个对象就成为垃圾。PHP会将其在内存中销毁;这是PHP的GC垃圾处理机制,防止内存溢出。当一个PHP线程结束时,当前占用的所有内存空间都会被销毁,当前程序中所有对象同时被销毁。GC进程一般都跟着每起一个SESSION而开始运行的。gc目的是为了在session文件过期以后自动销毁删除这些文件。

二、__destruct /unset

__destruct() 析构函数,是在垃圾对象被回收时执行。unset 销毁的是指向对象的变量,而不是这个对象。

三、 Session 与 GC

由于PHP的工作机制,它并没有一个daemon线程来定期的扫描Session信息并判断其是否失效,当一个有效的请求发生时,PHP 会根据全局变量 session.gc_probability和session.gc_divisor的值,来决定是否启用一个GC, 在默认情况下,session.gc_probability=1, session.gc_divisor =100也就是说有1%的可能性启动GC(也就是说100个请求中只有一个gc会伴随100个中的某个请求而启动)。

GC的工作就是扫描所有的Session信息,用当前时间减去session最后修改的时间,同session.gc_maxlifetime参数进行比较,如果生存时间超过gc_maxlifetime(默认24分钟),就将该session删除。但是,如果你Web服务器有多个站点,多个站点时,GC处理session可能会出现意想不到的结果,原因就是:GC在工作时,并不会区分不同站点的session。

那么这个时候怎么解决呢?

  1. 修改session.save_path,或使用session_save_path()让每个站点的session保存到一个专用目录。
  2. 提供GC的启动率,自然,GC的启动率提高,系统的性能也会相应减低,不推荐。
  3. 在代码中判断当前session的生存时间,利用session_destroy()删除。

看下面的例子:

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.

好,下一个:

<?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?

君请看:

<?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.

君再看:

<?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

君且看:

<?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 有更深刻的理解了。那么下面与之同工的例子自然好理解了。

<?php
error_reporting(E_ALL);
$a = ‘I am test.‘;
$b = & $a;
$b = null;
echo ‘$a = ‘. $a ."n";
echo ‘$b = ‘. $b ."n";
?>

下面我们来详细分析 GC 与引用:

所有例子中,创建了一个变量,这个过程通俗一点讲:是在内存中开辟了一块空间,在里面存放了一个字符串 I am test。 。 PHP 内部有个符号表,用来记录各块内存引用计数,那么此时会将这块内存的引用计数 加 1,并且用一个名为 $a 的标签(变量)指向这块内存,方便依标签名来操作内存。

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

在 Example 3 与 4 中,进行了 unset() 操作。根据实际的执行结果,可以看出: unset() 只是断开这个变量对它原先指向的内存的引用,使变量本身成为没有定义过空引用,所在调用时发出了 Notice ,并且使那块内存在符号表中引用计数 减 1,并没有影响到其他指向这块内存的变量。换言之,只有当一块内存在符号表中的引用计数为 0 时, PHP 引擎才会将这块内存回收。

看看下面的代码与其结果:

<?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() 的操作已经断开了指向,所以后继的操作不会对符号表的任何内存的引用记数造成影响了。

赋值 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 中直接将真实文件干掉了,所有指向它的软链接自然成了空链了。

之前在讨论到这些细节点时有很多想当然的念头,在实际的执行了测试代码后才发现: 哦,原来如此!纸上得来终觉浅,绝知此事须躬行。

时间: 2024-10-15 17:07:14

从几个简单的程序看PHP的垃圾回收机制的相关文章

Python程序运行流程与垃圾回收机制

Python程序运行流程 Python解释器首先将程序将py文件编译成一个字节码对象PyCodeObject(只存在于内存中).(当这个模块的 Python 代码执行完后,就会将编译结果保存到了pyc文件中,这样下次就不用编译,直接加载到内存中.pyc文件只是PyCodeObject对象在硬盘上的表现形式.) py文件被编译后,接下来的工作就交由 Python虚拟机来执行字节码指令.Python虚拟机会从编译得到的PyCodeObject对象中依次读入每一条字节码指令,并在当前的上下文环境中执行

java语言及其垃圾回收机制简单概述

 一.java 语言概述 Java 语言是一门纯粹的面向对象编程语言,它吸收了c++语言的各种优点.又摈弃了c++里难以理解的多继承,指针等概念因此Java语言具有功能强大和简单易用两个特征. Java语言的几个重要概念如下: J2ME:主要用于控制移动设备和信息家电等有限存储设备 J2SE:整个java技术的核心和基础, J2EE:java技术中应用最最广泛的部分,它提供了企业应用开发相关的完整的解决方案. API: 核心类库 JRE:运行Java程序所必须的环境的集合,包含JVM标准实现及J

部分 垃圾回收机制简短,简单的配置

垃圾回收机制(GC)它是JVM程序和算法对这些对象的释放不再使用的内存密集型. GC没有写java自定义标准语言,因此,不是所有的JVM有着GC. GC的主要目的就是清除不再使用的对象. 垃圾回收的两种方法: 1.引用计数 引用计数表示一个对象被引用的全部次数,当引用计数为0时,则表示该对象没有被引用,能够将其删除. 2.对象引用树 眼下比較经常使用的垃圾收集机制是对象引用树,即将对像的引用关系构建成一棵树,从一组根对象開始.对全部对象进行查找,通过递归查找若在该树中找到对应的对象,则将该对象标

编程语言类别;运行Python程序的方式;变量和常量;Python程序的垃圾回收机制;

---恢复内容开始--- 一.编程语言分类: 1.机器语言:直接用二进制的0和1和计算机(CPU)直接沟通交流,直接操作硬件. 优点:不需转换,计算机能直接读懂,执行速度快. 缺点:二进制代码复杂多样,理解甚难,开发效率低. ps:站在奴隶的角度说奴隶能够听得懂的话 0000 代表  加载 (LOAD) 0001代表 存储(STORE) 暂存器部分事例 0000  代表暂存器 A 0001  代表暂存器  B 存储器部分事例 000000000000  代表地址为 0的存储器 000000000

Java程序的垃圾回收机制

显示回收垃圾分析? 传统的C,C++语言 都是需要程序员负责回收已经分配的内存. 这个事情由程序员来做,会导致问题,因为程序员并不是总是知道内存该何时被释放,如果一些分配出的内存得不到及时回收,就会引起系统运行速度的下降,甚至程序瘫痪,这叫做内存泄漏. 缺点:1)程序忘记及时回收无用内存,从而导致内存泄漏,降低系统性能. 2)程序错误地回收系统核心的类库的内存,从而导致系统崩溃. Java程序的内存分配都是由Java运行环境JRE在后台自动进行的.JRE负责回收那些不再使用的内存,这种机制被称为

简单测试JVM垃圾回收机制 System.gc()方法

简单的写一个方法测试Java的垃圾回收机制 System.gc()可以提醒JVM虚拟机去进行垃圾回收了,但是不一定成功. /** * 覆写Object中的finalize()方法 */ public class LJ { @Override protected void finalize() throws Throwable { System.out.println("垃圾正在回收..........."); } } public class GCDemo { /** * @param

02 java 简单了解 垃圾回收机制

程序运行时,一些分配出去的内存得不到及时回收,就会引起系统运行速度下降,甚至导出程序瘫痪,这种现象被称为内存泄漏 java 程序里的 内存分配和回收都是由 jre 在后台自定进行的, jre 会负责回收那些不在使用的内存, 这种机制被称为 垃圾回收 Garbage Collection   --GC 通常 jre 会提供一个后台线程来进行检测和控制,一般都是在 CPU 空闲或者内存不足的时候进行 垃圾回收, 而陈序员 无法精确控制垃圾回收的时间和顺序 在java 中 当没有引用变量指向原先分配给

简单理解python的垃圾回收机制

关键词:垃圾回收.引用计数.分代回收.标记-清除 前言:理解python中变量的定义:抽象理解python中变量的定义过程 1.垃圾回收机制的基本组成: python采用的是以引用计数为主,以分代回收和标记清除为辅的垃圾回收机制 2.详细分析垃圾回收机制: (1)首先是引用计数: 在python中,每创建一个对象,那么python解释器会自动为其设置一个特殊的变量,这个变量称为引用计数(初始值默认是1).一旦有一个新变量指向这个对象,那么这个引用计数的值就会加1.如果引用计数的值为0.那么pyt

对CLR基本原理概念&amp;垃圾回收机制的简单理解

PS,之前有说过C语言的函数&变量的一些基本概念,说得可能不是很好,先也把C#的.里相关的也说下,已成一统. 而说函数变量,其实主要就是GC,而GC又是CLR的主要内容,故就有了此文. CLR基本原理: 把这几个概念一说基本就知道了,其实就是为了跨语言.跨平台,和JAVA的JVM类似 1.MSIL,中间语言,就是独立于所在平台系统的.net的特殊代码.里面含有相关元数据信息,常用的反编译工具想ILspy等就是靠他吃饭的. 2.CLR,公共语言运行时,其实就是把VB.VC.C#等不同语言编译成同一