我的编程空间,编程开发者的网络收藏夹
学习永远不晚

PHP中的引用计数是什么

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

PHP中的引用计数是什么

本篇内容主要讲解“PHP中的引用计数是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“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变量被引用赋值了,它的内存符号表服务于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会继续增加。然后unsetc 的引用赋值,可以看到 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 ,并不会影响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)=...// )

不管是对象还是数组,在打印调试时出现了 ... 这样的省略号,那么你的程序中就出现了循环引用。在之前的文章 关于PHP中对象复制的那点事儿 中我们也讲过这个循环引用的问题,所以这个问题应该是我们在日常开发中应该时刻关注的问题。

总结

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

测试代码:https://github.com/zhangyue0503/dev-blog/blob/master/php/202004/source/PHP%E7%9A%84%E5%BC%95%E7%94%A8%E8%AE%A1%E6%95%B0%E6%98%AF%E4%BB%80%E4%B9%88%E6%84%8F%E6%80%9D%EF%BC%9F.php参考文档:https://www.php.net/manual/zh/features.gc.refcounting-basics.phphttps://www.jianshu.com/p/52450a61354d

到此,相信大家对“PHP中的引用计数是什么”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

PHP中的引用计数是什么

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

下载Word文档

猜你喜欢

PHP中的引用计数是什么

本篇内容主要讲解“PHP中的引用计数是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“PHP中的引用计数是什么”吧!什么是引用计数在PHP的数据结构中,引用计数就是指每一个变量,除了保存了它们
2023-06-20

PHP中引用计数指的是什么意思

这篇文章将为大家详细讲解有关PHP中引用计数指的是什么意思,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。php是什么语言php,一个嵌套的缩写名称,是英文超级文本预处理语言(PHP:Hypertext P
2023-06-14

php中索引数组的作用是什么

这篇文章给大家介绍php中索引数组的作用是什么,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。PHP开发环境搭建工具有哪些一、phpStudy,是一个新手入门最常用的开发环境。二、WampServer,WampServe
2023-06-14

python中什么是引用计数器机制

本篇文章为大家展示了python中什么是引用计数器机制,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。python可以做什么Python是一种编程语言,内置了许多有效的工具,Python几乎无所不能,
2023-06-14

PHP中什么是索引数组

在PHP中,索引数组是指以数字为键名(下标)的数组。索引数组的下标(键名)由数字组成,默认从0开始,每个数字对应一个数组元素在数组中的位置,不需要特别指定,PHP会自动为索引数组的键名赋一个整数值,然后从这个值开始自动递增。本教程操作环境:windows7系统、PHP8.1版、DELL G3电脑数组是 PHP 中最重要的数据类型之一,在 PHP 中的应用非常广泛。因为 PHP 是弱数据类型的编程语言
2022-08-08

Python引用计数机制是什么

Python的引用计数机制是一种自动内存管理机制,用于跟踪对象的引用数量。每个对象都有一个引用计数,当一个对象被创建时,引用计数为1;每当一个新的引用指向该对象时,引用计数就会增加1;当一个引用被删除时,引用计数就会减少1。当一个对象的引用
2023-10-25

php中什么是引用传参

小编给大家分享一下php中什么是引用传参,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!什么是引用传参:在设定函数形参时,如果在形参变量之前添加符号,就表示实参传递
2023-06-15

PHP中索引数组的优点是什么

这篇文章给大家介绍PHP中索引数组的优点是什么,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。PHP开发环境搭建工具有哪些一、phpStudy,是一个新手入门最常用的开发环境。二、WampServer,WampServe
2023-06-14

PHP中引用符号(&)的作用是什么

本篇文章给大家分享的是有关PHP中引用符号(&)的作用是什么,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。按位与$a & $b 将把 $a 和 $b 中都为1的位设为1.奇偶数
2023-06-15

PHP7中zval结构和引用计数机制是什么

这篇文章主要介绍PHP7中zval结构和引用计数机制是什么,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!最近在查阅 PHP7 垃圾回收的资料的时候,网上的一些代码示例在本地环境下运行时出现了不同的结果,使我一度非常迷
2023-06-14

Python中怎么引用计数

Python中怎么引用计数,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。回顾内存地址Python中的任何变量都有对应的内存引用,也就是内存地址。如果不是容器类型,那么直接引
2023-06-16

java中什么是引用

引用的概念如果一个变量的类型是 类类型,而非基本类型,那么该变量又叫做引用。其实GC时主要看这个对象是否有引用指向该对象。按照这种引用的强弱的关系, 从JDK1.2版本开始,把对象的引用分为四种级别,从而使程序能更加灵活的控制对象的生命周期。这四种级别由高到低
java中什么是引用
2022-01-25

java中的引用是指什么

java中的类型有基本数据类型和复合类型。基本数据类型:如int,char等基本数据类型;复合类型:指针 和 引用;引用:为对象起立另一个名字,引用类型引用另一种类型。引用并非对象,相反的,它只是为一个已存在的对象所起的另外一个名字。如图1-1。引用必须进行初
java中的引用是指什么
2020-05-21

php中单双引号的区别是什么

在 php 中,单引号用于创建不进行转义或变量解析的简单字符串,而双引号则支持这些功能并允许创建复杂字符串。总体上,使用单引号的性能略优于双引号,因此对于简单字符串建议使用单引号。PHP 中单引号和双引号区别PHP 中单引号 (') 和双
php中单双引号的区别是什么
2024-04-29

PHP中引号转义的原理是什么

本篇文章给大家分享的是有关PHP中引号转义的原理是什么,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。PHP中有三个设置可以实现自动对’(单引号),”(双引号),\(
2023-06-17

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录