Zend引擎中實現了很多基本的數據結構,這些接口貫穿PHP和Zend引擎的始末,這些數據結構以及相應的操作接口都可以作為通用的接口來使用。本小節再簡單描述一下
在Zend引擎中HashTable的使用非常頻繁,這得益于他良好的查找性能,如果讀者看過前一小節會知道哈希表會預先分配內容以提高性能,而很多時候數據規模不會很大,固然使用哈希表能提高查詢性能,但是某些場景下并不會對數據進行隨機查找,這時使用哈希表就有點浪費了。
Zend引擎中的鏈表是[雙鏈表](http://zh.wikipedia.org/wiki/%E5%8F%8C%E9%93%BE%E8%A1%A8),通過雙鏈表的任意節點都能方便的對鏈表進行遍歷。
> Zend引擎的哈希表實現是哈希表和雙鏈表的混合實現,這也是為了方便哈希表的遍歷。
鏈表的實現很簡單,通常只需要三個關鍵元素:
1. 指向上個元素的指針
1. 指向下個元素的指針
1. 數據容器
Zend引擎的實現也很簡單,如下兩個是核心的數據接口,第一個是元素節點,第二個是鏈表容器。
typedef struct _zend_llist_element {
struct _zend_llist_element *next;
struct _zend_llist_element *prev;
char data[1]; /* Needs to always be last in the struct */
} zend_llist_element;
?
typedef struct _zend_llist {
zend_llist_element *head;
zend_llist_element *tail;
size_t count;
size_t size;
llist_dtor_func_t dtor;
unsigned char persistent;
zend_llist_element *traverse_ptr;
} zend_llist;
節點元素只含有前面提到的3個元素,第三個字段data和哈希表的實現一樣,是一個柔性結構體。

Zend zend_llist結構
如上圖所示,data字段的空間并不是只有一個字節,我們先看看元素插入的實現:
ZEND_API void zend_llist_add_element(zend_llist *l, void *element)
{
zend_llist_element *tmp = pemalloc(sizeof(zend_llist_element)+l->size-1, l->persistent);
?
tmp->prev = l->tail;
tmp->next = NULL;
if (l->tail) {
l->tail->next = tmp;
} else {
l->head = tmp;
}
l->tail = tmp;
memcpy(tmp->data, element, l->size);
?
++l->count;
}
如方法第一行所示,申請空間是額外申請了`l->size - 1`的空間。`l->size`是在鏈表創建時指定的,`zend_llist_element`結構體最后那個字段的注釋提到這個字段必須放到最后也是這個原因,例如curl擴展中的例子:`zend_llist_init(&(*ch)->to_free->slist, sizeof(struct curl_slist), (llist_dtor_func_t) curl_free_slist, 0);`, `size`指的是要插入元素的空間大小,這樣不同的鏈表就可以插入不同大小的元素了。
為了提高性能增加了鏈表頭和尾節點地址,以及鏈表中元素的個數。
最后的traverse_ptr 字段是為了方便在遍歷過程中記錄當前鏈表的內部指針,和哈希表中的:`Bucket *pInternalPointer;`字段一個作用。
### 操作接口
操作接口比較簡單,本文不打算介紹接口的使用,這里簡單說一下PHP源代碼中的一個小的約定,
如下為基本的鏈表遍歷操作接口:
/* traversal */
ZEND_API void *zend_llist_get_first_ex(zend_llist *l, zend_llist_position *pos);
ZEND_API void *zend_llist_get_last_ex(zend_llist *l, zend_llist_position *pos);
ZEND_API void *zend_llist_get_next_ex(zend_llist *l, zend_llist_position *pos);
ZEND_API void *zend_llist_get_prev_ex(zend_llist *l, zend_llist_position *pos);
?
#define zend_llist_get_first(l) zend_llist_get_first_ex(l, NULL)
#define zend_llist_get_last(l) zend_llist_get_last_ex(l, NULL)
#define zend_llist_get_next(l) zend_llist_get_next_ex(l, NULL)
#define zend_llist_get_prev(l) zend_llist_get_prev_ex(l, NULL)
一般情況下我們遍歷只需要使用后面的那組宏定義函數即可,如果不想要改變鏈表內部指針,可以主動傳遞當前指針所指向的位置。
PHP中很多的函數都會有`*_ex()`以及不帶ex兩個版本的函數,這主要是為了方便使用,和上面的代碼一樣,ex版本的通常是一個功能較全或者可選參數較多的版本,而在代碼中很多地方默認的參數值都一樣,為了方便使用,再封裝一個普通版本。
這里之所以使用宏而不是定義另一個函數是為了避免函數調用帶來的消耗,不過有的情況下還要進行其他的操作,也是會再定義一個新的函數的。
- 第一章 準備工作和背景知識
- 第一節 環境搭建
- 第二節 源碼結構、閱讀代碼方法
- 第三節 常用代碼
- 第四節 小結
- 第二章 用戶代碼的執行
- 第一節 生命周期和Zend引擎
- 第二節 SAPI概述
- Apache模塊
- 嵌入式
- FastCGI
- 第三節 PHP腳本的執行
- 詞法分析和語法分析
- opcode
- opcode處理函數查找
- 第四節 小結
- 第三章 變量及數據類型
- 第一節 變量的結構和類型
- 哈希表(HashTable)
- PHP的哈希表實現
- 鏈表簡介
- 第二節 常量
- 第三節 預定義變量
- 第四節 靜態變量
- 第五節 類型提示的實現
- 第六節 變量的生命周期
- 變量的賦值和銷毀
- 變量的作用域
- global語句
- 第七節 數據類型轉換
- 第八節 小結
- 第四章 函數的實現
- 第一節 函數的內部結構
- 函數的內部結構
- 函數間的轉換
- 第二節 函數的定義,傳參及返回值
- 函數的定義
- 函數的參數
- 函數的返回值
- 第三節 函數的調用和執行
- 第四節 匿名函數及閉包
- 第五節 小結
- 第五章 類和面向對象
- 第一節 類的結構和實現
- 第二節 類的成員變量及方法
- 第三節 訪問控制的實現
- 第四節 類的繼承,多態及抽象類
- 第五節 魔術方法,延遲綁定及靜態成員
- 第六節 PHP保留類及特殊類
- 第七節 對象
- 第八節 命名空間
- 第九節 標準類
- 第十節 小結
- 第六章 內存管理
- 第一節 內存管理概述
- 第二節 PHP中的內存管理
- 第三節 內存使用:申請和銷毀
- 第四節 垃圾回收
- 新的垃圾回收
- 第五節 內存管理中的緩存
- 第六節 寫時復制(Copy On Write)
- 第七節 內存泄漏
- 第八節 小結
- 第七章 Zend虛擬機
- 第一節 Zend虛擬機概述
- 第二節 語法的實現
- 詞法解析
- 語法分析
- 實現自己的語法
- 第三節 中間代碼的執行
- 第四節 PHP代碼的加密解密
- 第五節 小結
- 第八章 線程安全
- 第二節 線程,進程和并發
- 第三節 PHP中的線程安全
- 第九章 錯誤和異常處理
- 第十章 輸出緩沖
- 第十六章 PHP語言特性的實現
- 第一節 循環語句
- foreach的實現
- 第二十章 怎么樣系列(how to)
- 附錄
- 附錄A PHP及Zend API
- 附錄B PHP的歷史
- 附錄C VLD擴展使用指南
- 附錄D 怎樣為PHP貢獻
- 附錄E phpt測試文件說明
- 附錄F PHP5.4新功能升級解析
- 附錄G:re2c中文手冊