php的GC机制

在php5.3版本之前, php变量的回收机制只是简单的通过计数来处理(当refcount=0时,会回收内存),但这样会出现一个问题

$a=array("str");

$a[]=&$a;

unset($a);

执行unset之前,$a的refcount 为2,执行unset之后,$a的refcout为1,因为是1不等于0,不能被回收内存,即为垃圾,当然,在php脚本执行完毕后,所分配的内存将全部被回收,但是现在php除了应用于脚本以外,更多的地方用于写守护服务(当然我不推荐),可能长达一个月,两个月才结束脚本,这期间例如上面的程序会产生内存溢出

 注:unset并不能释放内存,需要看zval的refcount是否为0

php5.3以后增了GC垃圾回收机制

在分配zval时,以zval_gc_info为单位

 

#define ALLOC_ZVAL(z)                                     \do {                                                \(z) = (zval*)emalloc(sizeof(zval_gc_info));        \GC_ZVAL_INIT(z);                                \} while (0)typedef struct _zval_gc_info {zval z;union {gc_root_buffer       *buffered;struct _zval_gc_info *next;} u;
} zval_gc_info;#define FREE_ZVAL(z)                                     \do {                                                \GC_REMOVE_ZVAL_FROM_BUFFER(z);                    \efree(z);                                        \} while (0)

 

 

typedef struct _gc_root_buffer {struct _gc_root_buffer   *prev;        /* double-linked list               */struct _gc_root_buffer   *next;zend_object_handle        handle;    /* must be 0 for zval               */union {zval                 *pz;zend_object_handlers *handlers;} u;
} gc_root_buffer;

 

 

 

php的GC回收机制启动时,会分配10000个gc_root_buffer的空间

ZEND_API void gc_init(TSRMLS_D)  
{  if (GC_G(buf) == NULL && GC_G(gc_enabled)) {  GC_G(buf) = (gc_root_buffer*) malloc(sizeof(gc_root_buffer) * GC_ROOT_BUFFER_MAX_ENTRIES);  GC_G(last_unused) = &GC_G(buf)[GC_ROOT_BUFFER_MAX_ENTRIES];  gc_reset(TSRMLS_C);  }  
}  

 

在unset($a)时,详见这里 在active_systom_table找到key为a的zval后,对其value执行析构函数,将其zval的refcount-1,若减1后的值为0,说明可能直接释放内存,若为大于0,放到gc_root_buffer中

ZEND_API void _zval_ptr_dtor(zval **zval_ptr ZEND_FILE_LINE_DC) /* {{{ */  
{   Z_DELREF_PP(zval_ptr);  if (Z_REFCOUNT_PP(zval_ptr) == 0) {  TSRMLS_FETCH();  if (*zval_ptr != &EG(uninitialized_zval)) {  GC_REMOVE_ZVAL_FROM_BUFFER(*zval_ptr);  zval_dtor(*zval_ptr);  efree_rel(*zval_ptr);  }  } else {  TSRMLS_FETCH();  if (Z_REFCOUNT_PP(zval_ptr) == 1) {  Z_UNSET_ISREF_PP(zval_ptr);  }  GC_ZVAL_CHECK_POSSIBLE_ROOT(*zval_ptr);  }  
}  

 

 

 

#define GC_ZVAL_CHECK_POSSIBLE_ROOT(z) /gc_zval_check_possible_root((z) TSRMLS_CC)static zend_always_inline void gc_zval_check_possible_root(zval *z TSRMLS_DC)
{if (z->type == IS_ARRAY || z->type == IS_OBJECT) {gc_zval_possible_root(z TSRMLS_CC);}
}ZEND_API void gc_zval_possible_root(zval *zv TSRMLS_DC)
{if (UNEXPECTED(GC_G(free_list) != NULL &&GC_ZVAL_ADDRESS(zv) != NULL &&GC_ZVAL_GET_COLOR(zv) == GC_BLACK) &&(GC_ZVAL_ADDRESS(zv) < GC_G(buf) ||GC_ZVAL_ADDRESS(zv) >= GC_G(last_unused))) {/* The given zval is a garbage that is going to be deleted by* currently running GC */return;}if (zv->type == IS_OBJECT) {GC_ZOBJ_CHECK_POSSIBLE_ROOT(zv);return;}GC_BENCH_INC(zval_possible_root);//如果zv中的gc_root_buffer最后两位不是紫色,则进行处理if (GC_ZVAL_GET_COLOR(zv) != GC_PURPLE) {GC_ZVAL_SET_PURPLE(zv); //设置为紫色if (!GC_ZVAL_ADDRESS(zv)) {gc_root_buffer *newRoot = GC_G(unused);if (newRoot) {GC_G(unused) = newRoot->prev;} else if (GC_G(first_unused) != GC_G(last_unused)) {newRoot = GC_G(first_unused);GC_G(first_unused)++;} else {if (!GC_G(gc_enabled)) {GC_ZVAL_SET_BLACK(zv);return;}zv->refcount__gc++;gc_collect_cycles(TSRMLS_C);zv->refcount__gc--;newRoot = GC_G(unused);if (!newRoot) {return;}GC_ZVAL_SET_PURPLE(zv);GC_G(unused) = newRoot->prev;}newRoot->next = GC_G(roots).next;newRoot->prev = &GC_G(roots);GC_G(roots).next->prev = newRoot;GC_G(roots).next = newRoot;GC_ZVAL_SET_ADDRESS(zv, newRoot); //将gc_root_buffer放到zval_gc_info结构体中newRoot->handle = 0;newRoot->u.pz = zv;GC_BENCH_INC(zval_buffered);GC_BENCH_INC(root_buf_length);GC_BENCH_PEAK(root_buf_peak, root_buf_length);}}
}

 

 

将当前zval放到gc_root_bufer中,每个zval只放一次,依据是该zval所在的zval_gc_info中的gc_root_buffer的颜色 是否是 紫色

 

#define GC_ZVAL_GET_COLOR(v) \
GC_GET_COLOR(((zval_gc_info*)(v))->u.buffered)

#define GC_GET_COLOR(v) \
(((zend_uintptr_t)(v)) & GC_COLOR)

 

若不是紫色,则设置为紫色

#define GC_ZVAL_SET_PURPLE(v) \
GC_SET_PURPLE(((zval_gc_info*)(v))->u.buffered)

#define GC_SET_PURPLE(v) \
(v) = ((gc_root_buffer*)(((zend_uintptr_t)(v)) | GC_PURPLE))

 

宏GC_ZVAL_SET_ADDRESS(zv, newRoot);用来将gc_root_buffer newRoot 放到zv相应位置

#define GC_COLOR  0x03#define GC_BLACK  0x00
#define GC_WHITE  0x01
#define GC_GREY   0x02
#define GC_PURPLE 0x03#define GC_ZVAL_SET_ADDRESS(v, a) \GC_SET_ADDRESS(((zval_gc_info*)(v))->u.buffered, (a))#define GC_SET_ADDRESS(v, a) \(v) = ((gc_root_buffer*)((((zend_uintptr_t)(v)) & GC_COLOR) | ((zend_uintptr_t)(a))))

 

 

GC_ZVAL_SET_ADDRESS宏中先将v强制转为zval_gc_info类型,本身在为zval分配内存时, 就是以zval_gc_info为单位的,将强转为zval*,因为只需要对zval结构体填充数据,不需要 gc_root_buffer *bufer这个样成员
因为在结构体zval_gc_val中,zval z是第一个成员,那么z的地址也是zval_gc_info本身的内存地址

在PHP GC中,使用颜色来标明垃圾的处理过程

指针无论在32位机或64位机,最后两位均为0,

gc_collect_cycles处理垃圾

 


ZEND_API int gc_collect_cycles(TSRMLS_D)
{int count = 0;if (GC_G(roots).next != &GC_G(roots)) {zval_gc_info *p, *q, *orig_free_list, *orig_next_to_free;if (GC_G(gc_active)) {return 0;}GC_G(gc_runs)++;GC_G(zval_to_free) = FREE_LIST_END;GC_G(gc_active) = 1;gc_mark_roots(TSRMLS_C);gc_scan_roots(TSRMLS_C);gc_collect_roots(TSRMLS_C);orig_free_list = GC_G(free_list);orig_next_to_free = GC_G(next_to_free);p = GC_G(free_list) = GC_G(zval_to_free);GC_G(zval_to_free) = NULL;GC_G(gc_active) = 0;/* First call destructors */while (p != FREE_LIST_END) {if (Z_TYPE(p->z) == IS_OBJECT) {if (EG(objects_store).object_buckets &&EG(objects_store).object_buckets[Z_OBJ_HANDLE(p->z)].valid &&EG(objects_store).object_buckets[Z_OBJ_HANDLE(p->z)].bucket.obj.refcount <= 0 &&EG(objects_store).object_buckets[Z_OBJ_HANDLE(p->z)].bucket.obj.dtor &&!EG(objects_store).object_buckets[Z_OBJ_HANDLE(p->z)].destructor_called) {EG(objects_store).object_buckets[Z_OBJ_HANDLE(p->z)].destructor_called = 1;EG(objects_store).object_buckets[Z_OBJ_HANDLE(p->z)].bucket.obj.refcount++;EG(objects_store).object_buckets[Z_OBJ_HANDLE(p->z)].bucket.obj.dtor(EG(objects_store).object_buckets[Z_OBJ_HANDLE(p->z)].bucket.obj.object, Z_OBJ_HANDLE(p->z) TSRMLS_CC);EG(objects_store).object_buckets[Z_OBJ_HANDLE(p->z)].bucket.obj.refcount--;}}count++;p = p->u.next;}/* Destroy zvals */p = GC_G(free_list);while (p != FREE_LIST_END) {GC_G(next_to_free) = p->u.next;if (Z_TYPE(p->z) == IS_OBJECT) {if (EG(objects_store).object_buckets &&EG(objects_store).object_buckets[Z_OBJ_HANDLE(p->z)].valid &&EG(objects_store).object_buckets[Z_OBJ_HANDLE(p->z)].bucket.obj.refcount <= 0) {EG(objects_store).object_buckets[Z_OBJ_HANDLE(p->z)].bucket.obj.refcount = 1;Z_TYPE(p->z) = IS_NULL;zend_objects_store_del_ref_by_handle_ex(Z_OBJ_HANDLE(p->z), Z_OBJ_HT(p->z) TSRMLS_CC);}} else if (Z_TYPE(p->z) == IS_ARRAY) {Z_TYPE(p->z) = IS_NULL;zend_hash_destroy(Z_ARRVAL(p->z));FREE_HASHTABLE(Z_ARRVAL(p->z));} else {zval_dtor(&p->z);Z_TYPE(p->z) = IS_NULL;}p = GC_G(next_to_free);}/* Free zvals */p = GC_G(free_list);while (p != FREE_LIST_END) {q = p->u.next;FREE_ZVAL_EX(&p->z);p = q;}GC_G(collected) += count;GC_G(free_list) = orig_free_list;GC_G(next_to_free) = orig_next_to_free;}return count;
}

 

 
gc_mark_roots(TSRMLS_C); 为垃圾打下颜色标记,遍历gc_root_buffer,其中的u.pz将紫色变更为灰色,遍历u.pz该array zval的每个元素,将其refcount-1,
gc_collect_roots遍历gc_root_buffer,如果refcount==0再设置为白色,表示为垃圾,若refcount >0,表示别人在使用,设置为黑色
 
static void gc_mark_roots(TSRMLS_D)
{gc_root_buffer *current = GC_G(roots).next;while (current != &GC_G(roots)) {if (current->handle) {if (EG(objects_store).object_buckets) {//处理对象,暂时不用看}} else {if (GC_ZVAL_GET_COLOR(current->u.pz) == GC_PURPLE) {zval_mark_grey(current->u.pz TSRMLS_CC);} else {GC_ZVAL_SET_ADDRESS(current->u.pz, NULL);GC_REMOVE_FROM_BUFFER(current);}}current = current->next;}
}
 

 

 
static void zval_mark_grey(zval *pz TSRMLS_DC)
{Bucket *p;tail_call:if (GC_ZVAL_GET_COLOR(pz) != GC_GREY) {p = NULL;GC_BENCH_INC(zval_marked_grey);GC_ZVAL_SET_COLOR(pz, GC_GREY);if (Z_TYPE_P(pz) == IS_OBJECT && EG(objects_store).object_buckets) {//对象的处理, 暂时不用管} else if (Z_TYPE_P(pz) == IS_ARRAY) {if (Z_ARRVAL_P(pz) == &EG(symbol_table)) {GC_ZVAL_SET_BLACK(pz);} else {p = Z_ARRVAL_P(pz)->pListHead;}}while (p != NULL) {pz = *(zval**)p->pData;if (Z_TYPE_P(pz) != IS_ARRAY || Z_ARRVAL_P(pz) != &EG(symbol_table)) {pz->refcount__gc--;}if (p->pListNext == NULL) {goto tail_call;} else {zval_mark_grey(pz TSRMLS_CC);}p = p->pListNext;}}
}

 

 第二次遍历gc_root_buffer,如果zv中的颜色为灰色,且refcount=0,再置为白色,若refcount>0,置为黑色(不是垃圾)

static void gc_scan_roots(TSRMLS_D)
{gc_root_buffer *current = GC_G(roots).next;while (current != &GC_G(roots)) {if (current->handle) {zval z;INIT_PZVAL(&z);Z_OBJ_HANDLE(z) = current->handle;Z_OBJ_HT(z) = current->u.handlers;zobj_scan(&z TSRMLS_CC);} else {zval_scan(current->u.pz TSRMLS_CC);}current = current->next;}
}

 

 

static int zval_scan(zval *pz TSRMLS_DC)
{Bucket *p;tail_call:    if (GC_ZVAL_GET_COLOR(pz) == GC_GREY) {p = NULL;if (pz->refcount__gc > 0) {zval_scan_black(pz TSRMLS_CC);} else { GC_ZVAL_SET_COLOR(pz, GC_WHITE);if (Z_TYPE_P(pz) == IS_OBJECT && EG(objects_store).object_buckets) {//处理object} else if (Z_TYPE_P(pz) == IS_ARRAY) {if (Z_ARRVAL_P(pz) == &EG(symbol_table)) {GC_ZVAL_SET_BLACK(pz);} else {p = Z_ARRVAL_P(pz)->pListHead;}}}while (p != NULL) {if (p->pListNext == NULL) {pz = *(zval**)p->pData;goto tail_call;} else {zval_scan(*(zval**)p->pData TSRMLS_CC);}p = p->pListNext;}}return 0;
}

 

 

遍历gc_root_buffer链表,将zv颜色为白色的数据放置单独一个链表,全部回收

static void gc_collect_roots(TSRMLS_D)
{gc_root_buffer *current = GC_G(roots).next;while (current != &GC_G(roots)) {if (current->handle) {if (EG(objects_store).object_buckets) {struct _store_object *obj = &EG(objects_store).object_buckets[current->handle].bucket.obj;zval z;GC_SET_ADDRESS(obj->buffered, NULL);INIT_PZVAL(&z);Z_OBJ_HANDLE(z) = current->handle;Z_OBJ_HT(z) = current->u.handlers;zobj_collect_white(&z TSRMLS_CC);}} else {GC_ZVAL_SET_ADDRESS(current->u.pz, NULL);zval_collect_white(current->u.pz TSRMLS_CC);}GC_REMOVE_FROM_BUFFER(current);current = current->next;}
}

 

 

static void zval_collect_white(zval *pz TSRMLS_DC)
{Bucket *p;tail_call:if (((zval_gc_info*)(pz))->u.buffered == (gc_root_buffer*)GC_WHITE) {p = NULL;GC_ZVAL_SET_BLACK(pz);if (Z_TYPE_P(pz) == IS_OBJECT && EG(objects_store).object_buckets) {...} else {if (Z_TYPE_P(pz) == IS_ARRAY) {p = Z_ARRVAL_P(pz)->pListHead;}}/* restore refcount and put into list to free */pz->refcount__gc++;((zval_gc_info*)pz)->u.next = GC_G(zval_to_free);GC_G(zval_to_free) = (zval_gc_info*)pz;while (p != NULL) {pz = *(zval**)p->pData;if (Z_TYPE_P(pz) != IS_ARRAY || Z_ARRVAL_P(pz) != &EG(symbol_table)) {pz->refcount__gc++;}if (p->pListNext == NULL) {goto tail_call;} else {zval_collect_white(pz TSRMLS_CC);}p = p->pListNext;}}
}

 

 

 

转载于:https://www.cnblogs.com/taek/p/5453584.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/257216.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring 框架的JDBC模板技术

1. 概述 Spring 框架提供了很多持久层的模板类来简化编程;Spring 框架提供的JDBC模板类: JdbcTemplate 类;Spring 框架提供的整合 Hibernate 框架的模板类: HibernateTemplate 类2. 环境搭建 2.1 创建数据库表结构 CREATE TABLE t_account(id INT PRIMARY KEY AUTO_INCREMENT,…

BZOJ 1692: [Usaco2007 Dec]队列变换( 贪心 )

数据 n < 30000 , 然后 O( n ) 的贪心也过了..... USACO 数据是有多弱啊 ( ps : BZOJ 1640 和此题一模一样 , 双倍经验 ) --------------------------------------------------------------------------------------#include<cstdio>#include<cstring>#include…

数据说话,88000条数据绘制北京市地图

偶获得一批数据&#xff0c;本着好玩的态度绘制下来看看到底是什么鬼&#xff0c;绘制的结果如下&#xff1a; 呵呵&#xff0c;什么都不像。而且中间最重要的部分因数据量过大绘制的已经看不清楚了。于是乎&#xff0c;缩小绘制范围&#xff0c;去除周围没有用的数据。重新绘制…

我的第一个python web开发框架(11)——工具函数包说明(二)

db_helper.py是数据库操作包&#xff0c;主要有两个函数&#xff0c;分别是read()数据库读操作函数和write()数据库写操作函数。这个包的代码是从小戴同学分享的博文改造过来的。 1 #!/usr/bin/env python2 # codingutf-83 4 import psycopg25 from common import log_helper6 …

ASP.NET:在一般处理程序中通过 Session 保存验证码却无法显示图片?

1 using System.Drawing;2 using System.Web;3 using System.Web.SessionState;4 5 /// <summary>6 /// CaptchaHandler 的摘要说明7 /// </summary>8 public class CaptchaHandler : IHttpHandler, IRequiresSessionState  //简记&#xff1a;我需要Session9 { …

[LINK]用Python计算昨天、今天和明天的日期时间

用Python计算昨天、今天和明天的日期时间 转载于:https://www.cnblogs.com/Athrun/p/5477651.html

Windows系统下oracle数据库每天定时备份

第一步&#xff1a;建立备份脚本oraclebackup.bat 首先建立一个备份bat文件&#xff0c;在D盘下新建备份目录oraclebackup&#xff0c;将oracle安装目录下的EXP.EXE复制到此目录下&#xff0c;再新建一个文本文件oraclebackup.txt&#xff0c;内容如下&#xff1a; echo off ec…

面试题3:二维数组查找

1 bool Find(const int *matrix, int rows, int columns, int number)2 {3 int key;4 int indexRow;5 int indexCol;6 7 /*合法性检查*/8 if((NULL matrix)||(rows < 0)||(columns <0))9 { 10 return false; 11 } 12 13 /*提升…

linux crontab 命令

#method 1 crontab -e crontab -u root -e #不同用户自己的任务计划 crontab -l#method 2 vim /etc/crontab# Example of job definition: # .---------------- minute (0 - 59) # | .------------- hour (0 - 23) # | | .---------- day of month (1 - 31) # | | | .--…

[译] RNN 循环神经网络系列 2:文本分类

原文地址&#xff1a;RECURRENT NEURAL NETWORKS (RNN) – PART 2: TEXT CLASSIFICATION原文作者&#xff1a;GokuMohandas译文出自&#xff1a;掘金翻译计划本文永久链接&#xff1a;github.com/xitu/gold-m…译者&#xff1a;Changkun Ou校对者&#xff1a;yanqiangmiffy, To…

[置顶] Android开发者官方网站文档 - 国内踏得网镜像

Mark 一下&#xff1a; 镜像地址&#xff1a;http://wear.techbrood.com/index.html Android DevelopTools: http://www.androiddevtools.cn/ 转载于:https://www.cnblogs.com/superle/p/4561856.html

Java实现选择排序

选择排序思想就是选出最小或最大的数与第一个数交换&#xff0c;然后在剩下的数列中重复完成该动作。 package Sort;import java.util.Arrays;public class SelectionSort {public static int selectMinKey(int[] list, int beginIdx) {int idx beginIdx;int temp list[begin…

ASP.NET MVC中ViewData、ViewBag和TempData

1.ViewData 1.1 ViewData继承了IDictionary<string, object>,因此在设置ViewData属性时,传入key必须要字符串型别,value可以是任意类型。 1.2 ViewData它只会存在这次的HTTP要求而已,而不像Session可以将数据带到下HTTP要求。 public class TestController : Controller{…

java 正则表达式验证邮箱格式是否合规 以及 正则表达式元字符

package com.ykmimi.testtest; /*** 测试邮箱地址是否合规* author ukyor**/ public class EmailTest {public static void main(String[] args) {//定义要匹配的Email地址的正则表达式//其中\w代表可用作标识符的字符,不包括$. \w表示多个// \\.\\w表示点.后面有\w 括号{2,3}…

镜头选型

景深&#xff1a; 光圈越大&#xff0c;光圈值越小&#xff0c;景深越小 光圈越小&#xff0c;光圈值越大&#xff0c;景深越深 焦距越长&#xff0c;视角越小&#xff0c;主体像越大&#xff0c;景深越小 主体越近&#xff0c;景深越小

迅雷账号

账号 jiangchnangli:1 密码 892812 网址 http://www.s8song.net/read-htm-tid-4906661.html漫晴xydcq7681转载于:https://www.cnblogs.com/wlzhang/p/4563118.html

【Swift学习】Swift编程之旅---ARC(二十)

Swift使用自动引用计数(ARC)来跟踪并管理应用使用的内存。大部分情况下&#xff0c;这意味着在Swift语言中&#xff0c;内存管理"仍然工作"&#xff0c;不需要自己去考虑内存管理的事情。当实例不再被使用时&#xff0c;ARC会自动释放这些类的实例所占用的内存。然而…

像元大小及精度

说完了光学系统的分辨率之后我们来看看相机的图像分辨率。图像分辨率比较好理解&#xff0c;就是单位距离内的像用多少个像素来显示。以我们的ORCA-Flash4.0为例&#xff0c;芯片的像元大小为 6.5 μm&#xff0c;在 40X物镜的放大倍率下&#xff0c;1 μm的物经光学系统放大为…

转:传入的表格格式数据流(TDS)远程过程调用(RPC)协议流不正确 .

近期在做淘宝客的项目&#xff0c;大家都知道&#xff0c;淘宝的商品详细描述字符长度很大&#xff0c;所以就导致了今天出现了一个问题 VS的报错是这样子的 ” 传入的表格格式数据流(TDS)远程过程调用(RPC)协议流不正确“ 还说某个desricption 过长之类的话 直觉告诉我&#…

合并bin文件-----带boot发布版本比较好用的bat(便捷版)

直接上图上代码&#xff08;代码在结尾&#xff09;&#xff0c;有不会用的可以留言&#xff1a; 第一步&#xff1a;工程介绍&#xff0c;关键点--- 1.bat文件放所在app和boot工程的同级目录下 2.release为运行bat自动生成文件夹 第二步&#xff1a;合版.bat 针对具体项目需…