摘要:本文主要是針對,的話可以移步到慶哥的博客看,還有就是小菜我讀的是內(nèi)核剖析這本書。接下來我會使用到來調(diào)試源碼本文有參照博客中的部分內(nèi)容以及代碼。
前言
工作+實習快一年了,搞php后端開發(fā),一直很迷茫怎么提高自己,就先從php源碼開始吧,本人比較菜,本文章寫的比較趕時間,所以有什么錯誤或者漏掉的地方,望各位大神指正,多交流才能成長嘛,嘿嘿。
本文主要是針對php7,php5的話可以移步到慶哥的博客看,還有就是小菜我讀的是《php7內(nèi)核剖析》這本書。
接下來我會使用到xdebug來調(diào)試php源碼
本文有參照ohmygirl博客中的部分內(nèi)容以及代碼。
本文所用環(huán)境為windows,php7.0.10
php7中zval,zend_value的基本結(jié)構(gòu)php7和php5不同的地方有很多,zval,zend_value結(jié)構(gòu)就是其中之一
在php7中
zval定義在zend_types.h中
在zval這個結(jié)構(gòu)體重包含三個部分 zend_value(存儲實際的內(nèi)容),u1,u2兩個聯(lián)合體,其中u1主要存儲變量相關的一下屬性,而u2則是對u1的一些補充,例如當用到數(shù)組的時候,會用到u2.next來解決key哈希后出現(xiàn)的hash沖突
struct _zval_struct { zend_value value; /* 存儲變量的實際內(nèi)容 */ union { struct { ZEND_ENDIAN_LOHI_4( zend_uchar type, /* 存儲變量的類型 */ zend_uchar type_flags, /* 用于標識變量狀態(tài),例如GC方面的管理,通過設置為IS_TYPE_COLLECTABLE 則變量會被收集到GC中回收垃圾的buffer緩存區(qū)中 */ zend_uchar const_flags, zend_uchar reserved) /* call info for EX(This) */ } v; uint32_t type_info; } u1; union { uint32_t next; /* hash collision chain */ uint32_t cache_slot; /* literal cache slot */ uint32_t lineno; /* line number (for ast nodes) */ uint32_t num_args; /* arguments number for EX(This) */ uint32_t fe_pos; /* foreach position */ uint32_t fe_iter_idx; /* foreach iterator index */ uint32_t access_flags; /* class constant access flags */ uint32_t property_guard; /* single property guard */ uint32_t extra; /* not further specified */ } u2; };
zend_uchar type: 以下為外部使用的變量類型
#define IS_UNDEF 0 #define IS_NULL 1 #define IS_FALSE 2 #define IS_TRUE 3 #define IS_LONG 4 #define IS_DOUBLE 5 #define IS_STRING 6 #define IS_ARRAY 7 #define IS_OBJECT 8 #define IS_RESOURCE 9 #define IS_REFERENCE 10
php7中zend_value結(jié)構(gòu)
typedef union _zend_value { zend_long lval; /* long value */ double dval; /* double value */ zend_refcounted *counted; /*用于統(tǒng)計計數(shù)用,*/ zend_string *str; zend_array *arr; zend_object *obj; zend_resource *res; zend_reference *ref; zend_ast_ref *ast; zval *zv; void *ptr; zend_class_entry *ce; zend_function *func; struct { uint32_t w1; uint32_t w2; } ww; } zend_value;
這里我們先解釋一下php7的zval,zend_valu中重要的的幾個變量
zval中:
(1)zend_uchar type 這個是用來表示當前變量(例如 $a)是什么類型的變量($a是string類型?還是int類型?還是引用類型....)
zend_value中:
(1)zend_refcounted *counted; 表示引用計數(shù)的次數(shù), 何為引用計數(shù)? 就是zend_value變量被zval引用的次數(shù),例如我們$a=“abcs”;$c=$a;$b=$a;此時counted=3,如下圖,其中refcount也是實現(xiàn)GC自動內(nèi)存回收的基礎,下面會詳細講解php7變量的內(nèi)部實現(xiàn)
php7中對與變量的實現(xiàn)分以下幾種方式
(1).對于boolen類型,還有null,undefined,這種沒有具體值,只有類型的類型,直接在zval中通過zend_uchar type的類型來判斷,無需通過引用計數(shù)來實現(xiàn)。
正是因為沒有通過應用計數(shù)來實現(xiàn),所以它refcount為0
(2)對于int類型和float類型,因為在zend_value中有zend_long和double來保存數(shù)據(jù),如下圖,所以,在賦值的時候就不需要再使用引用計數(shù)了,在拷貝的直接進行賦值就行了,這樣做可以省掉大量引用計數(shù)的相關操作
定義一個$a=1,在php內(nèi)核中zval和zend_value的關系
(3)第三種就是常規(guī)的使用引用計數(shù)的方式來進行來進行變量的定義。在這些變量的實現(xiàn)中,都是通過指針指向一個具體的數(shù)據(jù)類型
例如 :
zend_string *str; struct _zend_string { zend_refcounted_h gc; zend_ulong h; /* hash value */ size_t len; char val[1]; };
$假設我定義一個$a="111";其內(nèi)部的實現(xiàn)就是,注意這里zend_string中char val[1],主要是因為C語言中字符串是以“0”結(jié)尾的,所以在zend_string中$a="111"這個變量值的存儲是char val[4]="111"。
1.普通賦值
前面說到,在php中,定義一個變量$a="444",實際上是生成了一個zval,和一個zend_value,然后zval指向這個zend_value來實現(xiàn)對$a="444"的定義的,然后通過refcount來統(tǒng)計引用的次數(shù)。
**所以可以總結(jié)出,refcount表示當前有多少個zval指向同一個zend_value**
我們定義如下:
$a="111";
$b=$a;
當然對于像boole型還有int,double,null變量,他們的直接通過zval保存,不會公用一個zend_value,所以直接使用深拷貝。
至于什么是深拷貝,什么是淺拷貝,最直接的區(qū)別就是在于有沒有重新生成一個一模一樣的zend_value,詳細請參看深拷貝和淺拷貝。
后面的寫時賦值(COW copy on write)就會使用到深拷貝。
對于php的賦值,實際上并不是所有的類型都是一樣的,剛剛也有說到,在php的zval中就有一個專門的字段用于標識當前類型適合哪種形式的那就是。
zend_uchar type_flags,
| refcounted | collectable | copyable | immutable ----------------+------------+-------------+----------+---------- simple types | | | | string | x | | x | interned string | | | | array | x | x | x | immutable array | | | | x object | x | x | | resource | x | | | reference | x | | |
zend_uchar type_flags這個字段用于標識當前的zval的屬于哪種賦值方式或者處于哪種狀態(tài),主要是用于內(nèi)存方面的管理具體參見內(nèi)存管理,
2.引用賦值
php中的引用賦值就是我們常用的引用,例如$a=&$b,這樣。
在php7中實現(xiàn)引用的時候,在php中實現(xiàn)引用的時候,會先生成一個zend_reference類型,這個類型中嵌套一個zval,然后這個zval的zend_value會指向之前的之前的那個zval的zend_value,原來的那個zval類型轉(zhuǎn)換成IS_REFERENCE類型,簡單來說**就是將原有的zval類型轉(zhuǎn)換成IS_REFERENCE,并新生成zend_reference類型指向原zend的zend_value。(好jb繞,理了好久)。
struct _zend_reference { zend_refcounted_h gc; zval val; };
例如我們定義一個$a="1111";$b=&$a;它的變換如下圖
$a="111"
$b=&$a;
可能有朋友注意到了,在zend_reference中refcount為2 ,但是在最后的真實的zend_value為refcount為1,這是為什么呢,其實很好理解,前面說過,refcount表示有多少個zval指向該zend_value,zend_reference中只有一個zval指向了zend_value。
中間加一層zend_reference這樣做其實有很多好處,這樣可以保留原有的zend_value類型不變,為深拷貝操作提供拷貝條件,下面我們舉個例子就知道了
是不是一目了然,只能說開發(fā)內(nèi)核的那些神牛太牛逼了。。。。。。。
當然關于php中值傳遞不僅僅那么簡單,還有很多很復雜的東西,小菜我也是才看沒多久,望各位大牛勿噴
php中的COW (copy on write)寫時復制是一種很重要的優(yōu)化手段,這里涉及到深拷貝和淺拷貝的知識,請移步。
關于深拷貝,剛剛上面的這個例子就是很好的說明
所謂深拷貝就是將原有的數(shù)據(jù)拷貝一份放到獨立分配一個地址空間。
而寫時賦值就是對深拷貝的一種優(yōu)化吧,意思是只有當發(fā)生寫操作的時候才進行深拷貝
舉個例子:
$a="3333"; $b=$a; $b.="444";
$a="3333";
$b=$a;
這個操作會使兩個zval指向同一個zend_value
這里并沒有觸發(fā)COW,執(zhí)行深拷貝
當$b.="444";發(fā)生了寫操作的時候,觸發(fā)COW,執(zhí)行深拷貝,拷貝了完全一樣的一份zend_value,$b所在的zval由原來的和$a所在的zval共同指向之前的zend_value, 轉(zhuǎn)換成指向拷貝出的新的zend_value
如果不進行深拷貝的話,那么當執(zhí)行$b.="444";后,$a也會等于“3333444”
當然不是所有的zend_value類型都可以進行復制,這個請參見我之前的那個zend_uchar type_flags表
**文章中可能有些不足的地方,懇求指正。
本來打算再寫寫GC回收的原理的,但是現(xiàn)在已經(jīng)2點多了,23333333.明天還要繼續(xù)打碼呢。。。。。。。不好意思**
下一篇準備寫一下php中的數(shù)組的實現(xiàn);
lift needs art,i need girl
http://bbs.csdn.net/topics/39...
http://blog.csdn.net/black_ox...
http://blog.csdn.net/xiaolei1...
https://segmentfault.com/a/11...
https://github.com/laruence/p...
https://www.cnblogs.com/ohmyg...
可能有遺漏的參考博客,望博主見諒
文章版權歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/28175.html
摘要:垃圾回收所謂垃圾就是指通過循環(huán)引用自己引用自己,目前只在類型中有出現(xiàn)的形式而導致永遠不為。當出現(xiàn)垃圾之后,的引擎有對應的垃圾回收機制。觸發(fā)這個機制的時機是每次出現(xiàn)減少時候。 自嘲)。。。。。2333,我覺得這是因為在php語言層面就幫我們解決了內(nèi)存回收的問題,但這讓我在和java大牛們吹牛逼的時候,聽到什么內(nèi)存泄露。。。。(納尼,我tmd怎么從來沒遇見過)一臉懵逼。 本人小菜,如果下面...
摘要:插入一個元素時先將元素按先后順序插入數(shù)組,位置是,再根據(jù)的哈希值映射到散列表中的某個位置,將存入這個位置查找時先在散列表中映射到,得到在數(shù)組的位置,再從數(shù)組中取出元素。目前只有兩種類型會使用這種機制。 1.變量結(jié)構(gòu) typedef struct _zval_struct zval; typedef union _zend_value { zend_long ...
摘要:告訴引擎要取的參數(shù)的信息,用來確保線程安全,返回值檢測是還是。數(shù)組遍歷假設我們需要一個取代以下功能的擴展的遍歷數(shù)組和差很多,提供了一些專門的宏來遍歷元素或。是一個關于線程安全的動作,用于避免各線程的作用域被其他的侵入。 起步 到這已經(jīng)能聲明簡單函數(shù),返回靜態(tài)或者動態(tài)值了。定義INI選項,聲明內(nèi)部數(shù)值或全局數(shù)值。本章節(jié)將介紹如何接收從調(diào)用腳本(php文件)傳入?yún)?shù)的數(shù)值,以及 PHP內(nèi)核...
摘要:操作數(shù)本身并無數(shù)據(jù)類型,它的數(shù)據(jù)類型由操作碼確定任何架構(gòu)的計算機都會對外提供指令集合運算器通過執(zhí)行指令直接發(fā)出控制信號控制計算機各項操作。 順風車運營研發(fā)團隊 李樂 1.從物理機說起 虛擬機也是計算機,設計思想和物理機有很多相似之處; 1.1馮諾依曼體系結(jié)構(gòu) 馮·諾依曼是當之無愧的數(shù)字計算機之父,當前計算機都采用的是馮諾依曼體系結(jié)構(gòu);設計思想主要包含以下幾個方面: 指令和數(shù)據(jù)不加區(qū)別...
摘要:中的用于類型整型和資源類型用于浮點類型用于字符串用于數(shù)組用于對象用于常量表達式才有多數(shù)文章,在提到變量結(jié)構(gòu)體的時候,都提到,實際上這個論述并不準確,在為時,這個結(jié)果是正確的。主要看中的,是兩個,這個永遠是個字節(jié),所以,因此。 PHP5 中的 zval // 1. zval typedef struct _zval_struct { zvalue_value value; ...
閱讀 2878·2021-10-14 09:43
閱讀 1672·2021-09-29 09:34
閱讀 1756·2021-07-28 00:16
閱讀 2971·2019-08-30 15:53
閱讀 2916·2019-08-30 13:59
閱讀 2971·2019-08-30 13:57
閱讀 1101·2019-08-26 13:38
閱讀 1906·2019-08-26 13:25