zoukankan      html  css  js  c++  java
  • php 浅谈垃圾回收机制

    php每创建一个变量,就会在zval中记录。一个zval变量容器,除了包含变量的类型和值,还包括两个字节的额外信息。

    第一个是"is_ref",用来标识这个变量是否是属于引用集合(reference set), bool类型。

    通过这个字节,php引擎才能把普通变量和引用变量区分开来,由于php允许用户通过使用&来使用自定义引用,zval变量容器中还有一个内部引用计数机制,来优化内存使用。

    第二个额外字节是"refcount",用以表示指向这个zval变量容器的变量(也称符号即symbol)个数。所有的符号存在一个符号表中,其中每个符号都有作用域(scope),那些主脚本(比如:通过浏览器请求的的脚本)和每个函数或者方法也都有作用域。

    安装Xdebug 来显示变量容器中的refcount 和 is_ref

    demo1

    <?php
    $a = "hello world";
    xdebug_debug_zval('a');
    输出:
    a:(refcount=1, is_ref=0)='hello world'

    新的变量a在当前作用域中生成了,同时生成了string类型 值为 ‘hello world’的变量容器。

    将一个变量赋值给另外一个变量时会增加引用次数。

    demo2

    <?php
    $a = "hello world";
    $b = $a;
    xdebug_debug_zval( 'a' );
    输出:
    a: (refcount=2, is_ref=0)='hello world'

    输出发现refcount变成了2,同一个变量容器被$a和$b关联。在非必要时,php是不会复制已经生成的变量容器的,当refcount=0时,变量容器就会被销毁。

    当任何关联到某个变量容器的变量离开它的作用域(函数执行结束)、或者对变量使用了unset()时,refcount就会-1

    demo3

    <?php
    $a = "hello world";
    $c = $b = $a;
    xdebug_debug_zval( 'a' );
    unset( $b, $c );
    xdebug_debug_zval( 'a' );
    输出:
    a: (refcount=3, is_ref=0)='hello world'
    a: (refcount=1, is_ref=0)='hello world'

    如果我们现在执行unset($a);,包含类型和值的这个变量容器就会从内存中删除。

    复合类型:

    demo4

    <?php
    $a = array( 'name' => 'life', 'num' => 11 );
    xdebug_debug_zval( 'a' );
    输出:
    a: (refcount=1, is_ref=0)=array (
       'name' => (refcount=1, is_ref=0)='life',
       'num' => (refcount=1, is_ref=0)=11
    )

    这三个zval变量容器是:a,name,num

    添加一个已经存在的元素到数组中:

    demo5

    <?php
    $a = array( 'name' => 'life', 'num' => 11 );
    $a['life'] = $a['name'];
    xdebug_debug_zval( 'a' );
    输出:
    a: (refcount=1, is_ref=0)=array (
       'name' => (refcount=2, is_ref=0)='life',
       'num' => (refcount=1, is_ref=0)=11,
       'life' => (refcount=2, is_ref=0)='life'
    )

    发现name和life的refcount都变成了2,但是life是新添加的,说明值为'life'的zval变量容器其实是同一个,xdebug_debug_zval()并没有显示这个信息。删除其中一个元素,另一个的refcount也会减少,废话不多说,上demo

    demo6

    <?php
    $a = array( 'name' => 'life', 'num' => 11 );
    $a['life'] = $a['name'];
    unset( $a['name'], $a['num'] );
    xdebug_debug_zval( 'a' );
    输出:
    a: (refcount=1, is_ref=0)=array (
       'life' => (refcount=1, is_ref=0)='life'
    )

    这样发现,一直都是refcount的值在改变,is_ref的值一直是false,当使用传引用的时is_ref的值才会发生变化。

    其他的回收周期或者性能方面的考虑,有时间再补上

  • 相关阅读:
    jQuery CVE-2019-11358原型污染漏洞分析和修复建议
    IIS中配置访问HTTPS
    如何把网址配置为http和https可以同时访问
    C#获取一周的工作日显示(星期几)
    Oracle查询数据库中所有表的记录数
    发布WebApi项目时,提示未包含binyourDocumentationFile.xml文档文件
    IntelliJ Idea 配置Tomcat提示Port is not specified
    C# 属性(Property)和字段(Field)的区别
    IDEA 出现错误:找不到或无法加载主类
    C# Enum 类型遍历
  • 原文地址:https://www.cnblogs.com/pfdltutu/p/9028604.html
Copyright © 2011-2022 走看看