PHP扩展开发与内核应用阅读笔记---php的内存管理
来源:互联网 发布:高中算法初步程序语言 编辑:程序博客网 时间:2024/05/17 09:48
PHP扩展开发与内核应用----第三章内存管理 阅读笔记!仅作为个人笔记,深入了解请移步:php扩展开发与内科应用
再次向作者表示感谢!同时欢迎同看此书的人加入QQ群:76761320
- 内存泄露
何为内存泄露:操作内存两个最基本的方法是申请内存,释放内存。如果应用程序向系统申请内存,系统便会在内存中寻找剩余的地方分配给应用程序,并标记下来,以后知道这块内存释放之前其它的应用程序都不能再用这块地方了。当这个应用程序不再使用申请的这块内存,却没有将这块内存释放,这样所有者应用程序也永远不再使用它了。那么我们就称之为内存泄露。- ZendMM
Zend内存管理(Zend Memory Manager,简称ZendMM、ZMM)层。ZendMM封装了系统的请求和释放内存的方法,为请求和释放内存提供了一个更加安全的方式。
所有这些在ZendMM中提供的内存管理函数都能够从下表中找到其在C语言中的函数。
C语言原生函数PHP内核封装后的函数void *malloc(size_t count);void *emalloc(size_t count);
void *pemalloc(size_t count, char persistent);void *calloc(size_t count);void *ecalloc(size_t count);
void *pecalloc(size_t count, char persistent);void *realloc(void *ptr, size_t count);void *erealloc(void *ptr, size_t count);
void *perealloc(void *ptr, size_t count, char persistent);void *strdup(void *ptr);void *estrdup(void *ptr);
void *pestrdup(void *ptr, char persistent);void free(void *ptr);void efree(void *ptr);
void pefree(void *ptr, char persistent);
- 重看zval
zval的四个成员value:变量的值type:变量当前的数据类型is_ref__gc:当一个变量被第一次创建的时候,它对应的zval结构体的is_ref_gc成员的值会被初始化为0,因为这个时候没有变量引用它,当有变量引用它的时候就会将is_ref_gc的值改为1。这时候表明这个zval结构是被引用的。
refcount__gc:当一个变量被第一次创建的时候,它对应的zval结构体的refcount__gc成员的值会被初始化为1,理由很简单,因为只有这个变量自己在用它。但是当你把这个变量赋值给别的变量时,refcount__gc属性便会加1变成2,因为现在有两个变量在用这个zval结构了!
写时复制机制
$a = 1;$b = $a;$b += 5;
第一句创建了一个zval结构,value=1,is_ref_gc=0,refcount__gc = 1。 第二句refcount__gc的值改为2。这时$a和$b是公用一个zval结构体的。 第三句内核首先查看refcount__gc属性,如果它大于1,并且is_ref_gc=0.则为这个变化的变量从原zval结构中复制出一份新的专属与$b的zval来,并改变其值。zval *get_var_and_separate(char *varname, int varname_len TSRMLS_DC){ zval **varval, *varcopy; if (zend_hash_find(EG(active_symbol_table),varname, varname_len + 1, (void**)&varval) == FAILURE) { /* 如果在符号表里找不到这个变量则直接return */ return NULL; } if ((*varval)->refcount < 2) { //如果这个变量的zval部分的refcount小于2,代表没有别的变量在用,return return *varval; } /* 如果这个zval在php语言中是通过引用的形式存在的,或者它的refcount小于2,则不许要复制。*/ if((*varval)->is_ref || (*varval)->refcount < 2) { return *varval; } /* 否则,复制一份zval*的值 */ MAKE_STD_ZVAL(varcopy); varcopy = *varval; /* 复制任何在zval*内已分配的结构*/ zval_copy_ctor(varcopy); /* 从符号表中删除原来的变量 * 这将减少该过程中varval的refcount的值 */ zend_hash_del(EG(active_symbol_table), varname, varname_len + 1); /* 初始化新的zval的refcount,并在符号表中重新添加此变量信息,并将其值与我们的新zval相关联。*/ varcopy->refcount = 1; varcopy->is_ref = 0; zend_hash_add(EG(active_symbol_table), varname, varname_len + 1,&varcopy, sizeof(zval*), NULL); /* 返回新zval的地址 */ return varcopy;}
Change on Write
$a = 1;$b = &$a;$b += 5;
第一句创建了一个zval结构,value=1,is_ref_gc=0,refcount__gc = 1。
第二句refcount__gc的值改为2,is_ref_gc=1。这时$a和$b是公用一个zval结构体的。 第三条语句,内核检查$b的zval当is_ref_gc为真,或者refcount__gc小于2的时候不需要复制值,直接更改值。下面是内核源码:/* 如果这个zval在php语言中是通过引用的形式存在的,或者它的refcount小于2,则不许要复制。*/if ((*varval)->is_ref || (*varval)->refcount < 2) { return *varval;}
PHP扩展开发与内核应用----第三章内存管理 阅读笔记!仅作为个人笔记,深入了解请移步:php扩展开发与内科应用
再次向作者表示感谢!同时欢迎同看此书的人加入QQ群:76761320
- 内存泄露
何为内存泄露:操作内存两个最基本的方法是申请内存,释放内存。如果应用程序向系统申请内存,系统便会在内存中寻找剩余的地方分配给应用程序,并标记下来,以后知道这块内存释放之前其它的应用程序都不能再用这块地方了。当这个应用程序不再使用申请的这块内存,却没有将这块内存释放,这样所有者应用程序也永远不再使用它了。那么我们就称之为内存泄露。
- ZendMM
Zend内存管理(Zend Memory Manager,简称ZendMM、ZMM)层。ZendMM封装了系统的请求和释放内存的方法,为请求和释放内存提供了一个更加安全的方式。
所有这些在ZendMM中提供的内存管理函数都能够从下表中找到其在C语言中的函数。
C语言原生函数PHP内核封装后的函数void *malloc(size_t count);void *emalloc(size_t count);void *pemalloc(size_t count, char persistent);void *calloc(size_t count);void *ecalloc(size_t count);
void *pecalloc(size_t count, char persistent);void *realloc(void *ptr, size_t count);void *erealloc(void *ptr, size_t count);
void *perealloc(void *ptr, size_t count, char persistent);void *strdup(void *ptr);void *estrdup(void *ptr);
void *pestrdup(void *ptr, char persistent);void free(void *ptr);void efree(void *ptr);
void pefree(void *ptr, char persistent);
- 重看zval
写时复制机制
$a = 1;$b = $a;$b += 5;
第一句创建了一个zval结构,value=1,is_ref_gc=0,refcount__gc = 1。
第二句refcount__gc的值改为2。这时$a和$b是公用一个zval结构体的。
zval *get_var_and_separate(char *varname, int varname_len TSRMLS_DC){ zval **varval, *varcopy; if (zend_hash_find(EG(active_symbol_table),varname, varname_len + 1, (void**)&varval) == FAILURE) { /* 如果在符号表里找不到这个变量则直接return */ return NULL; } if ((*varval)->refcount < 2) { //如果这个变量的zval部分的refcount小于2,代表没有别的变量在用,return return *varval; } /* 如果这个zval在php语言中是通过引用的形式存在的,或者它的refcount小于2,则不许要复制。*/ if((*varval)->is_ref || (*varval)->refcount < 2) { return *varval; } /* 否则,复制一份zval*的值 */ MAKE_STD_ZVAL(varcopy); varcopy = *varval; /* 复制任何在zval*内已分配的结构*/ zval_copy_ctor(varcopy); /* 从符号表中删除原来的变量 * 这将减少该过程中varval的refcount的值 */ zend_hash_del(EG(active_symbol_table), varname, varname_len + 1); /* 初始化新的zval的refcount,并在符号表中重新添加此变量信息,并将其值与我们的新zval相关联。*/ varcopy->refcount = 1; varcopy->is_ref = 0; zend_hash_add(EG(active_symbol_table), varname, varname_len + 1,&varcopy, sizeof(zval*), NULL); /* 返回新zval的地址 */ return varcopy;}
Change on Write
$a = 1;$b = &$a;$b += 5;
第一句创建了一个zval结构,value=1,is_ref_gc=0,refcount__gc = 1。
第二句refcount__gc的值改为2,is_ref_gc=1。这时$a和$b是公用一个zval结构体的。
/* 如果这个zval在php语言中是通过引用的形式存在的,或者它的refcount小于2,则不许要复制。*/if ((*varval)->is_ref || (*varval)->refcount < 2) { return *varval;}
- PHP扩展开发与内核应用阅读笔记---php的内存管理
- PHP扩展开发与内核应用阅读笔记---php的生命周
- PHP扩展开发与内核应用阅读笔记---php的作用域以及如何在扩展中定义,查找php变量
- PHP扩展开发与内核应用
- php扩展开发与内核应用
- 《PHP扩展开发及内核应用》
- PHP扩展开发及内核应用
- 《PHP扩展开发及内核应用》目录
- 《PHP扩展开发及内核应用》
- php内核探索笔记-内存的申请与释放
- php的扩展和嵌入--php内存管理
- PHP扩展开发笔记
- PHPER必读电子书推荐-《PHP扩展开发及内核应用》
- PHP内核--探究内存管理与缓存机制
- Linux 2.6 内核阅读笔记 内存管理
- PHP内核的学习--创建PHP扩展
- PHP 扩展开发笔记一
- php的内存管理
- 在50行代码贪吃蛇 笔记
- 构造函数不能声明为虚函数的原因是:
- 一个软件工程师的职业规划
- 黑马程序员_类加载器
- java 自学日志 【十六】---网络编程
- PHP扩展开发与内核应用阅读笔记---php的内存管理
- 关于winrar4.20注册码问题
- 黑马程序员_网络编程
- 你是狗吗?
- 黑马程序员_线程
- java.net.ConnectException: localhost/127.0.0.1:8080 - Connection refused
- JUnit的jar包和hamcrest的jar包关系
- 黑马程序员_异常
- MongoDB的初学三:MongoDB的基本命令