iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > PHP编程 >详解PHP的引用计数
  • 945
分享到

详解PHP的引用计数

2024-04-02 19:04:59 945人浏览 泡泡鱼
摘要

目录什么是引用计数怎么查看引用计数?对象的引用计数数组的引用计数关于内存泄露需要注意的地方总结什么是引用计数 在PHP的数据结构中,引用计数就是指每一个变量,除了保存了它们的类型和

什么是引用计数

PHP数据结构中,引用计数就是指每一个变量,除了保存了它们的类型和值之外,还额外保存了两个内容,一个是当前这个变量是否被引用,另一个是引用的次数。为什么要多保存这样两个内容呢?当然是为了垃圾回收(GC)。也就是说,当引用次数为0的时候,这个变量就没有再被使用了,就可以通过 GC 来进行回收,释放占用的内存资源。任何程序都不能无限制的一直占用着内存资源,过大的内存占用往往会带来一个严重的问题,那就是内存泄露,而 GC 就是php底层自动帮我们完成了内存的销毁,而不用像 C 一样必须去手动地 free 。

怎么查看引用计数?

我们需要安装 xdebug 扩展,然后使用 xdebug_debug_zval() 函数就可以看到指定内存的详细信息了,比如:


$a = "I am a String";
xdebug_debug_zval('a');
// a: (refcount=1, is_ref=0)='I am a String'

从上述内容中可以看出,这个 $a 变量的内容是 I am a String 这样一个字符串。而括号中的 refcount 就是引用次数,is_ref 则是说明这个变量是否被引用。我们通过变量赋值来看看这个两个参数是如何变化的。


$b = $a;
xdebug_debug_zval('a');
// a: (refcount=1, is_ref=0)='I am a String'

$b = &$a;
xdebug_debug_zval('a');
// a: (refcount=2, is_ref=1)='I am a String'

当我们进行普通赋值后,refcount 和 is_ref 没有任何变化,但当我们进行引用赋值后,可以看到 refcount 变成了2,is_ref 变成了1。这也就是说明当前的 \a 变量被引用赋值了,它的内存符号表服务于a变量被引用赋值了,它的内存符号表服务于a 和 $b 两个变量。


$c = &$a;
xdebug_debug_zval('a');
// a: (refcount=3, is_ref=1)='I am a String'

unset($c, $b);
xdebug_debug_zval('a');
// a: (refcount=1, is_ref=1)='I am a String'

$b = &$a;
$c = &$a;
$b = "I am a String new";
xdebug_debug_zval('a');
// a: (refcount=3, is_ref=1)='I am a String new'

unset($a);
xdebug_debug_zval('a');
// a: no such symbol

继续增加一个 c 的引用赋值,可以看到 refcount 会继续增加。然后 unset 掉c的引用赋值,可以看到refcount会继续增加。然后unset掉b 和 $c 之后,refcount 恢复到了1,不过这时需要注意的是,is_ref 依然还是1,也就是说,这个变量被引用过,这个 is_ref 就会变成1,即使引用的变量都已经 unset 掉了这个值依然不变。

最后我们 unset 掉 $a ,显示的就是 no such symbol 了。当前变量已经被销毁不是一个可以用的符号引用了。(注意,PHP中的变量对应的是内存的符号表,并不是真正的内存地址)

对象的引用计数

和普通类型的变量一样,对象变量也是使用同样的计数规则。


// 对象引用计数
class A{

}
$objA = new A();
xdebug_debug_zval('objA');
// objA: (refcount=1, is_ref=0)=class A {  }

$objB = $objA;
xdebug_debug_zval('objA');
// objA: (refcount=2, is_ref=0)=class A {  }

$objC = $objA;
xdebug_debug_zval('objA');
// objA: (refcount=3, is_ref=0)=class A {  }

unset($objB);
class C{

}
$objC = new C;
xdebug_debug_zval('objA');
// objA: (refcount=1, is_ref=0)=class A {  }

不过这里需要注意的是,对象的符号表是建立的连接,也就是说,对 objC 进行重新实例化或者修改为 NULL ,并不会影响objC进行重新实例化或者修改为NULL,并不会影响objA 的内容,对象进行普通赋值操作也是引用类型的符号表赋值,所以我们不需要加 & 符号。

数组的引用计数


// 数组引用计数
$arrA = [
    'a'=>1,
    'b'=>2,
];
xdebug_debug_zval('arrA');
// arrA: (refcount=2, is_ref=0)=array (
//     'a' => (refcount=0, is_ref=0)=1, 
//     'b' => (refcount=0, is_ref=0)=2
// )

$arrB = $arrA;
$arrC = $arrA;
xdebug_debug_zval('arrA');
// arrA: (refcount=4, is_ref=0)=array (
//     'a' => (refcount=0, is_ref=0)=1, 
//     'b' => (refcount=0, is_ref=0)=2
// )

unset($arrB);
$arrC = ['c'=>3];
xdebug_debug_zval('arrA');
// arrA: (refcount=2, is_ref=0)=array (
//     'a' => (refcount=0, is_ref=0)=1, 
//     'b' => (refcount=0, is_ref=0)=2
// )

// 添加一个已经存在的元素
$arrA['c'] = &$arrA['a'];
xdebug_debug_zval('arrA');
// arrA: (refcount=1, is_ref=0)=array (
//     'a' => (refcount=2, is_ref=1)=1, 
//     'b' => (refcount=0, is_ref=0)=2, 
//     'c' => (refcount=2, is_ref=1)=1
// )

调试数组的时候,我们会发现两个比较有意思的事情。

一是数组内部的每个元素又有单独的自己的引用计数。这也比较好理解,每一个数组元素都可以看做是一个单独的变量,但数组就是这堆变量的一个哈希集合。如果在对象中有成员变量的话,也是一样的效果。当数组中的某一个元素被 & 引用赋值给其他变量之后,这个元素的 refcount 会增加,不会影响整个数组的 refcount 。

二是数组默认上来的 refcount 是2。其实这是 PHP7 之后的一种新的特性,当数组定义并初始化后,会将这个数组转变成一个不可变数组(immutable array)。为了和普通数组区分开,这种数组的 refcount 是从2开始起步的。当我们修改一下这个数组中的任何元素后,这个数组就会变回普通数组,也就是 refcount 会变回1。这个大家可以自己尝试下,关于为什么要这样做的问题,官方的解释是为了效率,具体的原理可能还是需要深挖 PHP7 的源码才能知晓。

关于内存泄露需要注意的地方

其实 PHP 在底层已经帮我们做好了 GC 机制就不需要太关心变量的销毁释放问题,但是,千万要注意的是对象或数组中的元素是可以赋值为自身的,也就是说,给某个元素赋值一个自身的引用就变成了循环引用。那么这个对象就基本不太可能会被 GC 自动销毁了。


// 对象循环引用
class D{
    public $d;
}
$d = new D;
$d->d = $d;
xdebug_debug_zval('d');
// d: (refcount=2, is_ref=0)=class D { 
//     public $d = (refcount=2, is_ref=0)=... 
// }

// 数组循环引用
$arrA['arrA'] = &$arrA;
xdebug_debug_zval('arrA');
// arrA: (refcount=2, is_ref=1)=array (
//     'a' => (refcount=0, is_ref=0)=1, 
//     'b' => (refcount=0, is_ref=0)=2, 
//     'arrA' => (refcount=2, is_ref=1)=...
// )

不管是对象还是数组,在打印调试时出现了 ... 这样的省略号,那么你的程序中就出现了循环引用。所以这个问题应该是我们在日常开发中应该时刻关注的问题。

总结

引用计数是了解垃圾回收机制的前提条件,而且正是因为现代语言中都有一套类似的垃圾回收机制才让我们的编程变得更加容易且安全。那么有人说了,日常开发根本用不到这些呀?用不到不代表不应该去学习,就像循环引用这个问题一样,当代码中充斥着大量的类似代码时,系统崩溃只是迟早的事情,所以,这些知识是我们向更高级的程序进阶所不可或缺的内容。

测试代码: GitHub.com/zhangyue050…

以上就是详解PHP的引用计数的详细内容,更多关于PHP的引用计数的资料请关注编程网其它相关文章!

--结束END--

本文标题: 详解PHP的引用计数

本文链接: https://www.lsjlt.com/news/124812.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 详解PHP的引用计数
    目录什么是引用计数怎么查看引用计数?对象的引用计数数组的引用计数关于内存泄露需要注意的地方总结什么是引用计数 在PHP的数据结构中,引用计数就是指每一个变量,除了保存了它们的类型和...
    99+
    2024-04-02
  • PHP中的引用计数是什么
    本篇内容主要讲解“PHP中的引用计数是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“PHP中的引用计数是什么”吧!什么是引用计数在PHP的数据结构中,引用计数就是指每一个变量,除了保存了它们...
    99+
    2023-06-20
  • PHP中引用计数指的是什么意思
    这篇文章将为大家详细讲解有关PHP中引用计数指的是什么意思,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。php是什么语言php,一个嵌套的缩写名称,是英文超级文本预处理语言(PHP:Hypertext P...
    99+
    2023-06-14
  • 理解 JavaScript 内存管理中的引用计数
    引用计数是一种 JavaScript 内存管理技术,用于跟踪变量和对象对内存的引用次数。当引用计数为 0 时,表明不再有任何变量或对象引用该内存,因此可以安全地将其释放。 工作原理: 每个 JavaScript 变量都维护一个引用计数器...
    99+
    2024-04-02
  • SQLServer索引设计基础知识详解使用
    目录一、前言二、索引设计背景知识2.1、索引设计策略包括的任务三、常规索引设计3.1、数据库注意事项3.2、查询注意事项3.3、列注意事项3.4、索引的特征3.5、索引排序顺序设计指...
    99+
    2023-05-14
    SQL Server索引设计 SQL索引设计
  • php如何引用计数实现垃圾回收
    这篇文章主要为大家展示了“php如何引用计数实现垃圾回收”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“php如何引用计数实现垃圾回收”这篇文章吧。1、实现说明给对象添加引用计数器,每次在某个地方...
    99+
    2023-06-15
  • PHP中单引号和双引号的区别详解
    目录PHP中单引号和双引号简介PHP中字符串和变量插值时单引号和双引号的区别PHP中转义撇号时单引号和双引号的区别补充知识:字符串内部如果出现PHP引号怎么办--关于转义.总结本文将...
    99+
    2023-01-15
    php单引号和双引号区别在哪 php单双引号的区别 php中双引号和单引号
  • Windows 容器中 PHP 的索引使用方法详解
    在 Windows 容器中使用 PHP 时,索引是一个非常重要的概念。索引是一种数据结构,用于快速查找和访问数组中的元素。在 PHP 中,数组是一种非常常见的数据类型,因此索引的使用非常重要。本文将详细介绍在 Windows 容器中使用 ...
    99+
    2023-10-04
    windows 容器 索引
  • PHP程序员必备的数组索引技巧:path和numy索引的使用方法详解!
    数组是PHP中最常用的数据结构之一,它可以存储多个值,并且可以通过键名来访问这些值。在PHP中,有许多种不同的数组索引技巧,但是其中两种最常用的是path和numy索引。在本篇文章中,我们将详细讲解这两种索引的使用方法以及它们的优缺点。 ...
    99+
    2023-10-01
    path numy 索引
  • SQL Server索引设计基础知识详解使用
    目录一、前言二、索引设计背景知识2.1、索引设计策略包括的任务三、常规索引设计3.1、数据库注意事项3.2、查询注意事项3.3、列注意事项3.4、索引的特征3.5、索引排序顺序设计指南总结一、前言 索引设计不佳和缺少索引...
    99+
    2023-04-03
    SQL Server索引设计 SQL索引设计
  • php代码审计,php漏洞详解
    文章目录 1、输入验证和输出显示2、命令注入(Command Injection)3、eval 注入(Eval Injection)4、跨网站脚本攻击(Cross Site Scripting...
    99+
    2023-09-02
    php chrome 开发语言
  • Python计数器collections.Counter用法详解
    目录一. 介绍二. 基本操作1. 统计“可迭代序列”中每个元素的出现的次数2. most_common()统计出现次数最多的元素3. elements() 和...
    99+
    2023-03-07
    Python计数器collections.Counter Python collections.Counter
  • 详解PHP中数组函数的巧用
    目录前言取指定键名移除指定键名数组去重重置索引清除空值确认数组成员全部为真获取指定键名之前/之后的数组数组中重复次数最多的值前言 PHP 的数组是一种很强大的数据类型,与此同时 PH...
    99+
    2024-04-02
  • PHP如何带索引检查计算数组的交集,用回调函数比较索引
    这篇文章将为大家详细讲解有关PHP如何带索引检查计算数组的交集,用回调函数比较索引,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。PHP使用带索引检查回调函数计算数组交集 介绍 在PHP中,计算数组交集是常...
    99+
    2024-04-02
  • C++引用的详细解释
    目录一、C++ 引用1.规则2.应用3.引用提高1.可以定义指针的引用,但不能定义引用的引用。2.可以定义指针的指针,不能定义引用的指针。3.可以定义指针数组,但不能定义引用数组,可...
    99+
    2024-04-02
  • Python中如何引用计数
    Python中如何引用计数,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。变量是内存引用Python中的变量是内存引用。如果输入x = [1,2]时会发生什么[1...
    99+
    2023-06-16
  • Golang 函数设计模式的应用详解
    go语言函数式编程模式包括:命令模式:将操作封装成对象,实现请求延迟。策略模式:使用函数作为策略,动态更改算法。回调函数:作为参数传递给其他函数,灵活控制流程。这些模式通过函数作为一等公...
    99+
    2024-04-19
    golang 函数设计模式 go语言
  • PythonNumPy数组索引的示例详解
    目录前言1、访问数组元素2、访问 2-D Arrays(数组)3、访问 3-D Arrays(数组)4、负索引前言 NumPy(Numerical Python的缩写)是一个开源的P...
    99+
    2023-01-18
    Python NumPy 数组索引 Python NumPy 索引 NumPy 数组索引
  • Python中怎么引用计数
    Python中怎么引用计数,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。回顾内存地址Python中的任何变量都有对应的内存引用,也就是内存地址。如果不是容器类型,那么直接引...
    99+
    2023-06-16
  • PHP的文件索引同步技术详解
    PHP作为一种常用的服务器端编程语言,经常需要对文件进行操作。在文件操作中,文件索引同步技术是非常重要的。本文将详细介绍PHP的文件索引同步技术,同时提供相应的演示代码。 一、什么是文件索引同步技术? 在文件操作中,文件索引是指将文件的关...
    99+
    2023-07-09
    同步 文件 索引
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作