0

0

变量改变时PHP内核做了些什么?

php中文网

php中文网

发布时间:2016-06-13 08:54:24

|

1119人浏览过

|

来源于php中文网

原创

变量改变时PHP内核做了些什么?

看下面的内容之前先对zval这个结构体做个了解

  1. typedef struct _zval_struct {
  2. zvalue_value value;
  3. zend_uint refcount;
  4. zend_uchar type;
  5. zend_uchar is_ref;
  6. } zval;

zval结构体中共有4个元素,value是一个联合体,用来真正的存储zval的值,refcount用来计数该zval被多少个变量使用,type表示zval所存储的数据类型,is_ref用来标志该zval是否被引用。

引用计数

  1. $a = 'Hello World';
  2. $b = $a;
  3. unset($a);
  4. >

我们一起来剖析下上面这段代码:

  • $a = 'Hello World';首先这句代码被执行,内核创建一个变量,并分配12字节的内存去存储字符串'Hello World'和末尾的NULL。

  • $b = $a;接着执行这句代码,执行这句的时候内核里面发生了什么呢?

    • $a所指向的zval中的refcount进行加1操作。

      立即学习PHP免费学习笔记(深入)”;

    • 将变量$b指向$a所指向的zval。
      在内核中大概是这样的,其中active_symbol_table是当前的变量符号表

      1. {
      2. zval *helloval;
      3. MAKE_STD_ZVAL(helloval);
      4. ZVAL_STRING(helloval, "Hello World", 1);
      5. zend_hash_add(EG(active_symbol_table), "a", sizeof("a"),
      6. &helloval, sizeof(zval*), NULL);
      7. ZVAL_ADDREF(helloval);
      8. zend_hash_add(EG(active_symbol_table), "b", sizeof("b"),
      9. &helloval, sizeof(zval*), NULL);
      10. }
  • unset($a);这句代码执行后,内核会将azvalrefcountb还和原来一样

写时复制

  1. $a = 1;
  2. $b = $a;
  3. $b += 5;
  4. ?>

上面这段代码执行完之后,一般肯定希望$a=1,$b=6,但是如果像引用计数那样,$a$b指向相同的zval,修改$b之后$a不是也变了?
这个具体是怎么实现的呢,我们一起来看下:

  • $a = 1;内核创建一个zval,并分配4个字节存储数字1。

  • $b = $a;这一步和引用计数中的第二步一样,将$b指向和$a相同的zval,并将zval中的引用计数值refcount加1。

  • $b += 5;关键是这一步,这一步骤发生了什么呢,怎么确保修改之后不影响$a

    • 其实Zend内核在改变zval之前都会去进行get_var_and_separete操作,如果recfount>1,就需要分离就创建新的zval返回,否则直接返回变量所指向的zval,下面看看如何分离产生新的zval。

    • 复制一个和$b所指向zval一样的zval。

    • $b所指向的zval中的refcount计数减1。

      北极象沉浸式AI翻译
      北极象沉浸式AI翻译

      免费的北极象沉浸式AI翻译 - 带您走进沉浸式AI的双语对照体验

      下载
    • 初始化生成的新zval,设置refcount=1,is_ref=0。

    • $b指向新生成的zval。

    • 对新生成的zval进行操作,这就是写时复制。
      下面看看内核中分离时的主要代码:

      1. zval *get_var_and_separate(char *varname, int varname_len TSRMLS_DC)
      2. {
      3. zval **varval, *varcopy;
      4. if (zend_hash_find(EG(active_symbol_table),
      5. varname, varname_len + 1, (void**)&varval) == FAILURE) {
      6. /* Variable doesn't actually exist fail out */
      7. return NULL;
      8. }
      9. if ((*varval)->is_ref || (*varval)->refcount < 2) {
      10. /* varname is the only actual reference,
      11. * or it's a full reference to other variables
      12. * either way: no separating to be done
      13. */
      14. return *varval;
      15. }
      16. /* Otherwise, make a copy of the zval* value */
      17. MAKE_STD_ZVAL(varcopy);
      18. varcopy = *varval;
      19. /* Duplicate any allocated structures within the zval* */
      20. zval_copy_ctor(varcopy);
      21. /* Remove the old version of varname
      22. * This will decrease the refcount of varval in the process
      23. */
      24. zend_hash_del(EG(active_symbol_table), varname, varname_len + 1);
      25. /* Initialize the reference count of the
      26. * newly created value and attach it to
      27. * the varname variable
      28. */
      29. varcopy->refcount = 1;
      30. varcopy->is_ref = 0;
      31. zend_hash_add(EG(active_symbol_table), varname, varname_len + 1,
      32. &varcopy, sizeof(zval*), NULL);
      33. /* Return the new zval* */
      34. return varcopy;
      35. }

写时改变

  1. $a = 1;
  2. $b = &$a;
  3. $b += 5;
  4. ?>

上面这段代码执行完之后一般希望是:$a == $b == 1。这个又是怎么实现的呢?

  • $a = 1;这一步骤和写时复制中的第一步一样。

  • $b = &$a;这一步骤内核会将$b指向$a所指向的zval,将zval中的refcount加1,并将zval中的is_ref置为1。

  • $b += 5;这一步骤和写时复制中的第三步骤一样,但是内核中发生的事情却不一样。

    • 内核看到$b进行变化的时候,也会执行get_var_and_separate函数,看是否需要分离。

    • 如果(*varval)->is_ref的话也会直接返回$b所指向的zval,不去分离产生新的zval,不管zval的refcount是否>1。

    • 这时候再去修改$b值,$a的值也就改变了,因为他们指向相同的zval。

分离的问题

说道现在聪明的你可能已经看出点问题了,如果一个zval结构体既有refcount计数又有is_ref引用这个时候怎么办?

  1. $a = 1;
  2. $b = $a;
  3. $c = &$a;
  4. ?>

如果出现上面这种情况的时候,如果$a、$b、$c指向同一个zval结构体,进行改变的时候Zend到底去听谁的?其实这个地方不会指向同一个zval了。
如果对一个is_ref = 0 && refcount >1的zval进行写时改变这种赋值形式(就是引用赋值)的时候,Zend会将等号右边的变量分离出来一个新的zval,
对这个zval进行初始化,对之前的zval的refcount进行减1操作,让等号左边的变量指向这个新的zval,refcount进行加1操作,is_ref=1。看看下面这张图片


  1. $a = 1;
  2. $b = &$a;
  3. $c = $a;
  4. ?>

上面这又是另外一种情况,在is_ref = 1的情况下,试图单纯的进行refcount+1操作的时候会分离出来一个新的zval给等号左边的变量,并初始化他,看看下面这张图片

参考文献

1.《Extending and Embedding PHP》- Chaper 3 - Memory Management.



PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

相关标签:

php

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
苹果官网入口直接访问
苹果官网入口直接访问

苹果官网直接访问入口是https://www.apple.com/cn/,该页面具备0.8秒首屏渲染、HTTP/3与Brotli加速、WebP+AVIF双格式图片、免登录浏览全参数等特性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

115

2025.12.24

拼豆图纸在线生成器
拼豆图纸在线生成器

拼豆图纸生成器有PixelBeads在线版、BeadGen和“豆图快转”;推荐通过pixelbeads.online或搜索“beadgen free online”直达官网,避开需注册的诱导页面。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

82

2025.12.24

俄罗斯搜索引擎yandex官方入口地址(最新版)
俄罗斯搜索引擎yandex官方入口地址(最新版)

Yandex官方入口网址是https://yandex.com。用户可通过网页端直连或移动端浏览器直接访问,无需登录即可使用搜索、图片、新闻、地图等全部基础功能,并支持多语种检索与静态资源精准筛选。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

546

2025.12.24

JavaScript ES6新特性
JavaScript ES6新特性

ES6是JavaScript的根本性升级,引入let/const实现块级作用域、箭头函数解决this绑定问题、解构赋值与模板字符串简化数据处理、对象简写与模块化提升代码可读性与组织性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

150

2025.12.24

php框架基础知识汇总
php框架基础知识汇总

php框架是构建web应用程序的架构,提供工具和功能,以简化开发过程。选择合适的框架取决于项目需求和技能水平。实战案例展示了使用laravel构建博客的步骤,包括安装、创建模型、定义路由、编写控制器和呈现视图。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

20

2025.12.24

Word 字间距调整方法汇总
Word 字间距调整方法汇总

本专题整合了Word字间距调整方法,阅读下面的文章了解更详细操作。

47

2025.12.24

任务管理器教程
任务管理器教程

本专题整合了任务管理器相关教程,阅读下面的文章了解更多详细操作。

7

2025.12.24

AppleID格式
AppleID格式

本专题整合了AppleID相关内容,阅读专题下面的文章了解更多详细教程。

12

2025.12.24

csgo视频观看入口合集
csgo视频观看入口合集

本专题整合了csgo观看入口合集,阅读下面的文章了知道更多入口地址。

371

2025.12.24

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 7.8万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 6.9万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号