资讯专栏INFORMATION COLUMN

PHP底层分析:关于写时复制(cow)

widuu / 779人阅读

摘要:可以看到,该结构体存储了关于变量值,有几个变量指向该结构体,变量类型,是否为引用变量等信息。这个就是写时复制,在作怪,他没有在赋值的时候就分裂成两个结构体,而是在我们改写其中一个变量时发生效果,属于一种慢复制也称慢分裂。

想要走到技术的天花板,那么学习过程中在于知其然且知其所以然。

今天我们来讨论一下PHP底层的写时复制(也称写时分裂)。

首先我们先来看看一段代码:

两段代码的输出结果相信各位都知道,但是我们今天讲讲这之中发生了什么。

下图是PHP存储变量的结构体(为方便讲解已写了注释),zend.h在Zend目录下。

可以看到,该结构体存储了关于变量值,有几个变量指向该结构体,变量类型,是否为引用变量等信息。

那么第一次打印发生了什么呢?变量的信息进入了一个结构体,相关如下:

$name = ‘傍晚八点半’;
$myName = $name;

此时$name和$myName共用一个结构体的,refcount__gc为2,

我们发现,$myName = $name;这个过程中并没有主动变成两个结构体(这也算PHP内部实现优化的一种,只用一个结构体,省了内存)。

那么当代码运行到 $myName = ‘gzchen’; 的时候,结构体如何变化呢?由于第一次输出时是两个变量共用结构体,那么此时更改其中一个变量,会不会导致两个值一起变化呢?纯粹从结构体的逻辑来看,是有可能的,毕竟大家共用着这个结构体嘛。

那么我们看下第二次打印是怎么样的情况,相关变化如下:

并没有按照我们所想的将$name和$myName同时改成’gzchen’,而是复制多了一份结构体出来,两个结构体分别对应着$name和$myName。

这个就是写时复制(Copy-on-write,COW)在作怪,他没有在$myName = $name;赋值的时候就分裂成两个结构体,而是在我们改写其中一个变量时发生效果,属于一种慢复制(也称慢分裂)。

伪代码如下:

我们再看下另外一段代码:

输出为’b’,中途发生了什么?

其实foreach遍历过程中,并不是直接操作$arr(原数组)的,而是会将$arr复制出一个$arrcopy(实际上是一个副本,我这里以$arrcopy代替),foreach在遍历过程中操作的其实一直是$arrcopy,大概的流程是这样:

和上面举得例子其实是一个道理,我们可以看出,刚开始($arr = $arrcopy)还是共用一个结构体的,但是$arr[$k] = $v又再次赋值,发生了写时复制,结构体就分裂了。

然后前面说过foreach操作的是$arrcopy,所以$arr的结构体指针就被停留在第一位了(因为结构体不一样了,$arrcopy没办法同步给$arr赋值了)。

其实这类技术通常只会在面试中用到,日常开发会用这种写法的人终究还是少数,暂时看不明白的朋友也不用太在意,只要知道有”写时复制”这个情况出现就行了。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/21525.html

相关文章

  • PHP底层分析关于强制分裂

    摘要:前面写过一篇底层分析关于写时复制今天来讲讲关于强制分裂的知识,简单来说,强制分裂就是在引用变量主动赋值前,该变量传值赋值过,就会发生强制分裂。实际开发基本用不到这层原理,但在面试中强制分裂通常会和写时复制一起考。 学习需要知其然而知其所以然,PHP底层相关就是这类知识。 前面写过一篇《PHP底层分析:关于写时复制(cow)》:https://segmentfault.com/a/119...

    刘明 评论0 收藏0
  • foreach遍历过程中的奇怪现象(PHP5)

    摘要:中基础中的三大坑,遍历,引用机制,数组。今天我们在讲讲中的一些奇怪现象。本文适合有一定基础的。运行流程共用一个结构体开始遍历数组,进行判断,拷贝数组是一个新的结构体,操作的是新的结构体。那么遍历数组时,全程与原数组无关。 PHP中基础中的三大坑,foreach遍历,引用机制&,数组。 今天我们在讲讲foreach中的一些奇怪现象。 在讲解之前,可以先看看我其他相关的文章,属于同一个大的...

    kgbook 评论0 收藏0
  • PHP_底层分析

    摘要:将会产生强制分裂结构体结构体引用数组时的一些奇怪现象引用数组时的怪现象数组不会比较细致的检查,多维数组存在。因此,判断的时候,只会判断外面一层的结构体。中底层都离不开表。底层所有的变量都是放在中。 PHP编译特点 编译型语言 对于C语言,C++,编译成机器码(二进制)来运行。Java语言,把.java 编译成.class, 称为bytecode(字节码),由jvm来运行 解释型语言 解...

    tomlingtm 评论0 收藏0
  • PHP写时复制(Copy On Write)

    摘要:只有在真正需要使用资源时才占用资源,写时复制通常能减少资源的占用。基础方面规范新特性性能调优垃圾回收机制安全攻击原理和防范攻击原理和防范注入攻击防范密码哈希计算机网络协议协议连接过程 从一个例子说起: 很明显在这段代码执行以后,$var_dup 的值应该还是laruence, 那么这又是怎么实现的呢?这就是 PHP 的 copy on write 机制: PHP 在修改一个变量以前,...

    arashicage 评论0 收藏0
  • php7内核阅读(1)--数据容器zval和zend_value

    摘要:本文主要是针对,的话可以移步到庆哥的博客看,还有就是小菜我读的是内核剖析这本书。接下来我会使用到来调试源码本文有参照博客中的部分内容以及代码。 前言 工作+实习快一年了,搞php后端开发,一直很迷茫怎么提高自己,就先从php源码开始吧,本人比较菜,本文章写的比较赶时间,所以有什么错误或者漏掉的地方,望各位大神指正,多交流才能成长嘛,嘿嘿。本文主要是针对php7,php5的话可以移步到庆...

    canger 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<