今回は「PHP8の基盤となるカーネルソースコードの解析 - 配列(1)」を紹介します。一定の参考値があるので、困っている友達が参考になれば幸いです。
おすすめ関連記事: 「PHP8 の基礎となるカーネル ソース コードの分析 - 配列 (2) 」 「PHP8 の基礎となるカーネル ソース コードの分析 - 配列 (3) " PHP8 の基礎となるカーネル ソース コードの分析 - 配列 (4) >>
PHP 配列は変数 (zval の配列ポインター型) で使用されるだけでなく、シンボルテーブルなどのカーネル。
PHP8 では、配列は _zendarray を使用して zend_array と hashtable
のエイリアスを表します。エイリアスが 2 つある理由は、以前の下位バージョンの関数との互換性のためです (一部は今見ることができます) 関数またはマクロ コードで、ハッシュテーブルを使用するものと、zend_array を使用するものがあることがわかります)
PHP の配列は、「双方向に順序付けされた多次元リンク リスト」
1 つは 2 つの特徴があります。キーと値のペア
#2 を保存します。 Ordered
高度なハッシュテーブルとして理解できます
PHP8における配列の定義はzend_types.hにあり、コアコードは以下の通りです
typedef struct _zend_array zend_array; //别名zend_array typedef struct _zend_array HashTable; //别名 HashTable struct _zend_array { zend_refcounted_h gc; //和zend_string一样 还记得前面的zend_string 吗? /// gc 占用8个字节 用于引用计数和 字符串类型的记录 union { struct { ZEND_ENDIAN_LOHI_4( zend_uchar flags, // flags 8位的无符号字符, 最大值为255 标记HashTable用 PHP8 中有6个值 //#define HASH_FLAG_CONSISTENCY ((1<<0) | (1<<1)) //#define HASH_FLAG_PACKED (1<<2) //#define HASH_FLAG_UNINITIALIZED (1<<3) //#define HASH_FLAG_STATIC_KEYS (1<<4) /* long and interned strings */ //#define HASH_FLAG_HAS_EMPTY_IND (1<<5) //#define HASH_FLAG_ALLOW_COW_VIOLATION (1<<6) zend_uchar _unused, zend_uchar nIteratorsCount, //迭代器计数。foreach语句会在全局变量EG中创建一个迭代器, //迭代器包含正在遍历的HashTable和游标信息。 //nIteratorsCount记录了当前runtime正在迭代当前HashTable的迭代器的数量。 zend_uchar _unused2) } v; //这里有点不一样 看陈雷大佬书中 v结构体还包括 u.v.nApplyCount和u.v.consistency uint32_t flags; // } u; // u是是一个联合体。占用4个字节。 //可以存储一个uint32_t类型的flags,也可以存储由4个unsigned char组成的结构体v, //这里的宏ZEND_ENDIAN_LOHI_4是为了兼容不同操作系统的大小端,可以忽略。 Bucket *arData; //HashTable中存储数据的单元的指针。 // 用来存储key和value以及辅助信息的容器。 uint32_t nTableSize; // HashTable的大小。表示arData指向的bucket数组的大小,即所有bucket的数量。 //该字段取值始终是2n,最小值是8,最大值在64位系统中是0x80000000(2的31次幂)。 uint32_t nNumUsed; //指所有已使用bucket的数量,包括有效bucket和无效bucket的数量 uint32_t nNumOfElements; //有效bucket的数量。该值总是小于或等于nNumUsed uint32_t nTableMask; //标记。一般值为 -nTableSize。 uint32_t nInternalPointer; //全局默认游标。reset/key/current/next/prev等宏 和操作都会用到 zend_long nNextFreeElement; //下一个插入的元素的key的下标 //比如 当$a[] = 1 nNextFreeElement =1 dtor_func_t pDestructor; //指向一个函数 typedef void (*dtor_func_t)(zval *pDest); //可以看出是pDest是zval结构指针二级指针, //为什么会是二级指针,因为c语言函数传递都是值传递,要改变指针值只能将指针地址传入 //当bucket元素被更新或者被删除时,会对bucket的value调用该函数, //如果value是引用计数的类型,那么会对value引用计数减1,进而引发可能的gc。 }; typedef struct _zend_refcounted_h { uint32_t refcount;//一个 32位纯数字的 refcount uint32_t type_info; } u; } zend_refcounted_h; //_zend_refcounted_h // 包括 一个 32位纯数字的 refcount 和一个联合体u //联合体u里面包括一个 type_infozend_refcounted_h 占用8字节,refount英文翻译成中文是引用的意思 显然 这个 zend_refcounted_h是为了引用计数和字符串类别存储用的。 typedef struct _Bucket { zval val; //数组的值 ( 复习下 zval只有16个字节) zend_ulong h; // key的 hash值 zend_string *key; //数组的key的 指针 /* string key or NULL for numerics */ } Bucket;
▏この記事は、原著者 PHP Cui Xuefeng の同意を得て、PHP 中国語 Web サイト、原文アドレス: https://zhuanlan.zhihu.com/p/352830733
に掲載されています。以上がPHP8 の基礎となるカーネル ソース コードの解析 - 配列 (1)の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。