摘要:所以想要理解更深入的同學(xué)最好查看下我之前的關(guān)于介紹變量函數(shù)的文章類的數(shù)據(jù)結(jié)構(gòu)不管是普通類還是抽象類或是接口,都存放到統(tǒng)一的結(jié)構(gòu)體中,并且在生成中間代碼時,會將此類添加到全局類列表中。
對于PHPer來說,OOP是不可或缺的開發(fā)思維,但是你對php類和對象的底層實現(xiàn)又了解多少呢?本著知其然且知其所以然的思想,讓我們一起來尋找答案~
類的底層實現(xiàn)可看作是之前我們講過的變量、函數(shù)等的知識集合。所以想要理解更深入的同學(xué)最好查看下我之前的關(guān)于介紹變量、函數(shù)的文章
類的數(shù)據(jù)結(jié)構(gòu)不管是普通類還是抽象類或是接口,都存放到統(tǒng)一的結(jié)構(gòu)體中,并且在生成中間代碼時,會將此類添加到全局類列表中。當(dāng)然,也是在此時,會通過類名判斷該類是否已經(jīng)存在,如果存在,則添加失敗
struct _zend_class_entry { char type; // 和函數(shù)一樣,類被拆分為兩種類型:ZEND_INTERNAL_CLASS 內(nèi)部類型和ZEND_USER_CLASS 用戶自定義類型 char *name;// 類名稱 zend_uint name_length; // 即sizeof(name) - 1 struct _zend_class_entry *parent; // 繼承的父類 int refcount; // 引用數(shù) zend_bool constants_updated; zend_uint ce_flags; //類的類型,在編譯階段被區(qū)分是普通類,接口,抽象類 HashTable function_table; // 靜態(tài)類方法和普通類方法存放集合 HashTable default_properties; // 默認屬性存放集合 HashTable properties_info; // 屬性信息存放集合 HashTable default_static_members;// 類本身所具有的靜態(tài)變量存放集合 HashTable *static_members; // type == ZEND_USER_CLASS時,取&default_static_members; // type == ZEND_INTERAL_CLASS時,設(shè)為NULL HashTable constants_table; // 常量存放集合 struct _zend_function_entry *builtin_functions;// 方法定義入口 /* 魔術(shù)方法 */ //所有魔術(shù)方法多帶帶存放,初始化時被設(shè)置為null union _zend_function *constructor; union _zend_function *destructor; union _zend_function *clone; union _zend_function *__get; union _zend_function *__set; union _zend_function *__unset; union _zend_function *__isset; union _zend_function *__call; union _zend_function *__tostring; union _zend_function *serialize_func; union _zend_function *unserialize_func; zend_class_iterator_funcs iterator_funcs;// 迭代 /* 類句柄 */ zend_object_value (*create_object)(zend_class_entry *class_type TSRMLS_DC); zend_object_iterator *(*get_iterator)(zend_class_entry *ce, zval *object, intby_ref TSRMLS_DC); /* 類聲明的接口 */ int(*interface_gets_implemented)(zend_class_entry *iface, zend_class_entry *class_type TSRMLS_DC); /* 序列化回調(diào)函數(shù)指針 */ int(*serialize)(zval *object, unsignedchar**buffer, zend_uint *buf_len, zend_serialize_data *data TSRMLS_DC); int(*unserialize)(zval **object, zend_class_entry *ce, constunsignedchar*buf, zend_uint buf_len, zend_unserialize_data *data TSRMLS_DC); zend_class_entry **interfaces; // 類實現(xiàn)的接口 zend_uint num_interfaces; // 類實現(xiàn)的接口數(shù) char *filename; // 類的存放文件地址 絕對地址 zend_uint line_start; // 類定義的開始行 zend_uint line_end; // 類定義的結(jié)束行 char *doc_comment; zend_uint doc_comment_len; struct _zend_module_entry *module; // 類所在的模塊入口:EG(current_module) };
由上面代碼可以看出,類的成員變量、成員方法都是存放在各自的結(jié)構(gòu)體中,而結(jié)構(gòu)體的數(shù)據(jù)結(jié)構(gòu)和之前講解的變量和函數(shù)的數(shù)據(jù)結(jié)構(gòu)一模一樣,只不過編譯后的成員變量和成員方法是存放在類結(jié)構(gòu)體中而已
對象的生成我們都知道,對象是new出來的,但是從底層來看,對象生成分為3步
第一步:根據(jù)類名去全局類列表內(nèi)查找該類是否存在,如果存在,則獲取存儲類的變量
第二步:判斷類是否是普通類(非抽象類或接口);如果是普通類則給需要創(chuàng)建的對象存放的zval容器分配內(nèi)存,并設(shè)置容器類型為IS_OBJECT
第三步:執(zhí)行對象初始化操作,將對象添加到全局對象列表(對象池)中
附上對象的數(shù)據(jù)結(jié)構(gòu):
typedef struct _zend_object { zend_class_entry *ce; //對象的類結(jié)構(gòu) HashTable *properties; //對象屬性 HashTable *guards; /* protects from __get/__set ... recursion */ } zend_object;獲取和設(shè)置成員變量
獲取成員變量:
第一步,獲取對象的屬性,從對象的properties查找是否存在與名稱對應(yīng)的屬性,如果存在返回結(jié)果,如果不存在,轉(zhuǎn)第二步
第二步,如果存在get魔術(shù)方法,則調(diào)用此方法獲取變量,如果不存在,則報錯
設(shè)置成員變量:
第一步,獲取對象的屬性,從對象的properties查找是否存在與名稱對應(yīng)的屬性,如果存在且已有的值和需要設(shè)置的值相同,則不執(zhí)行任何操作,否則執(zhí)行變量賦值操作,如果不存在,轉(zhuǎn)第二步
第二步,如果存在_set魔術(shù)方法,則調(diào)用此方法設(shè)置變量,如果不存在,轉(zhuǎn)第三步
第三步,如果成員變量一直沒有被設(shè)置過,則直接將此變量添加到對象的properties字段所在HashTable中。
到今天為止,我們差不多已經(jīng)將關(guān)于php的底層原理講了一個遍了。當(dāng)然,在這期間,不少同學(xué)跟我說,現(xiàn)在都已經(jīng)逐漸開始使用php7了,你現(xiàn)在講解的內(nèi)容還是php5,會不會過時了?其實我講解php5也是為講php7作準(zhǔn)備,php7畢竟是php5的延展,了解了php5之后,再去了解php7會更加容易些。而且php也是從php5開始才逐漸完善起來的,我們有必要了解下php5的內(nèi)容。不過從下周開始,我們會開始從底層比較php7和php5的不同,敬請期待~
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/31281.html
摘要:第一階段基礎(chǔ)階段基礎(chǔ)程序員重點把搞熟練核心是安裝配置基本操作目標(biāo)能夠完成基本的系統(tǒng)安裝,簡單配置維護能夠做基本的簡單系統(tǒng)的開發(fā)能夠在中型系統(tǒng)中支持某個功能模塊的開發(fā)。本項不做重點學(xué)習(xí),除非對前端有興趣。 第一階段:基礎(chǔ)階段(基礎(chǔ)PHP程序員) 重點:把LNMP搞熟練(核心是安裝配置基本操作) 目標(biāo):能夠完成基本的LNMP系統(tǒng)安裝,簡單配置維護;能夠做基本的簡單系統(tǒng)的PHP開發(fā);能夠在P...
摘要:寫在前面深入系列共計篇已經(jīng)正式完結(jié),這是一個旨在幫助大家,其實也是幫助自己捋順底層知識的系列。深入系列自月日發(fā)布第一篇文章,到月日發(fā)布最后一篇,感謝各位朋友的收藏點贊,鼓勵指正。 寫在前面 JavaScript 深入系列共計 15 篇已經(jīng)正式完結(jié),這是一個旨在幫助大家,其實也是幫助自己捋順 JavaScript 底層知識的系列。重點講解了如原型、作用域、執(zhí)行上下文、變量對象、this、...
摘要:對于來說,變量有全局變量和局部變量之分那么,他們都是存儲到一個哈希表內(nèi)了么其實不是的,變量存儲也有作用域的概念。 上次跟大家講了垃圾回收機制后,有些小伙伴對底層原理比較感興趣,私信問我了一些關(guān)于變量的相關(guān)知識,既然大家對變量比較感興趣,那么這次我們來系統(tǒng)的講一下變量的底層原理 變量結(jié)構(gòu) 首先,我們還是先擺上我們的zval結(jié)構(gòu)體,即php所有變量都會以zval結(jié)構(gòu)體的形式實現(xiàn) struc...
本文是公眾號讀者jianfeng投稿的面試經(jīng)驗恭喜該同學(xué)成功轉(zhuǎn)型目錄:毅然轉(zhuǎn)型,沒頭蒼蠅制定目標(biāo),系統(tǒng)學(xué)習(xí)面試經(jīng)歷毅然轉(zhuǎn)崗,沒頭蒼蠅首先,介紹一下我的背景。本人坐標(biāo)廣州,2016年畢業(yè)于一個普通二本大學(xué),曾經(jīng)在某機構(gòu)培訓(xùn)過Android。2018年初的時候已經(jīng)在兩家小公司工作干了兩年的android開發(fā),然后會一些Tomcat、Servlet之類的技術(shù),當(dāng)時的年薪大概也就15萬這樣子。由于個人發(fā)展...
閱讀 1173·2021-11-22 15:22
閱讀 3847·2021-10-19 13:13
閱讀 3596·2021-10-08 10:05
閱讀 3305·2021-09-26 10:20
閱讀 2993·2019-08-29 14:21
閱讀 2203·2019-08-27 10:55
閱讀 1879·2019-08-26 10:31
閱讀 2588·2019-08-23 16:47