PHP HASH表
在PHP中,所有的数据 无论变量,常量,类,属性 都用Hash表来实现.
先要说说 HASH表
[c]
typedef struct bucket {
ulong h; /* Used for numeric indexing */
uint nKeyLength; //key长度
void *pData; //指向 Bucke保存的数据 指针
void *pDataPtr; //指针数据
struct bucket *pListNext; //下一个元素指针
struct bucket *pListLast;//上一个元素指针
struct bucket *pNext;
struct bucket *pLast;
char arKey[1]; /* Must be last element */
} Bucket;
typedef struct _hashtable {
uint nTableSize;//HashTable的大小
uint nTableMask;//等于nTableSize-1
uint nNumOfElements;//对象个数
ulong nNextFreeElement;//指向下一个空元素位置 nTableSize+1
Bucket *pInternalPointer; /* Used for element traversal *///保存当前遍历的指针
Bucket *pListHead;//头元素指针
Bucket *pListTail;//尾元素指针
Bucket **arBuckets;//存储hash数组数据
dtor_func_t pDestructor;//类似于析构函数
zend_bool persistent;//用哪种方法分配内存空间 PHP统一管理内存还是用普通的malloc
unsigned char nApplyCount;//当前hash bucket被访问的次数,是否遍历过数据,防止无限递归循环
zend_bool bApplyProtection;
#if ZEND_DEBUG
int inconsistent;
#endif
} HashTable;
[/c]
我们结合 HASH表初始化函数来说
[c]
ZEND_API int _zend_hash_init(HashTable *ht, uint nSize, hash_func_t pHashFunction, dtor_func_t pDestructor, zend_bool persistent ZEND_FILE_LINE_DC)
{
uint i = 3;
Bucket **tmp;
SET_INCONSISTENT(HT_OK);
if (nSize >= 0x80000000) { //HASH表大小大于0x8则初始化为0x8
/* prevent overflow */
ht->nTableSize = 0x80000000;
} else {
while ((1U << i) < nSize) { //调整为 2的n次方 i++; } ht->nTableSize = 1 << i;//HASH bucket大小 为 2的i次方 i=3 ,nTableSize最小值为8
}
//为了提高计算效率,系统自动会将nTableSize调整到最小一个不小于nTableSize的2的整数次方。也就是说,如果在初始化HashTable时指定一个nTableSize不是2的整数次方,系统将会自动调整nTableSize的值
ht->nTableMask = ht->nTableSize - 1;
ht->pDestructor = pDestructor;//一个函数指针,当HashTable发生增,删,改时调用
ht->arBuckets = NULL;
ht->pListHead = NULL;
ht->pListTail = NULL;
ht->nNumOfElements = 0;
ht->nNextFreeElement = 0;
ht->pInternalPointer = NULL;
ht->persistent = persistent;//如果persisient为TRUE,则使用操作系统本身的内存分配函数为Bucket分配内存,否则使用PHP的内存分配函数
ht->nApplyCount = 0;
ht->bApplyProtection = 1;
/* Uses ecalloc() so that Bucket* == NULL */
if (persistent) { //操作系统本身内存分配方式分配内存,calloc分配内存后自动初始化为0
tmp = (Bucket **) calloc(ht->nTableSize, sizeof(Bucket *));
if (!tmp) {
return FAILURE;
}
ht->arBuckets = tmp;
} else {//用PHP的内存管理机制分配内存
tmp = (Bucket **) ecalloc_rel(ht->nTableSize, sizeof(Bucket *));
if (tmp) {
ht->arBuckets = tmp;
}
}
//自动申请一块内存给arBuckets,该内存大小等于 nTableSize
return SUCCESS;
}
[/c]
-
在读源码的时候 ,经常会看到 EG,PG,CG这样的宏
CG是compile_global的简写
EG是excutor_global的简写
G就是全局变量的意思
我们就以EG宏为例:
[c]
#ifdef ZTS
# define EG(v) TSRMG(executor_globals_id, zend_executor_globals *, v)
#else
# define EG(v) (executor_globals.v)
extern ZEND_API zend_executor_globals executor_globals;
#endif
[/c]
很简单 只是一个获取全局变量的宏
那么我们看看 zend_executor_globals这个结构体
在/Zend/zend.h里面定义
typedef struct _zend_executor_globals zend_executor_globals;
是一个 _zend_executor_globals的别名
同一个文件里找到它
PHP的所有 局部变量,全局变量,函数,类的 Hash表 都在这里定义了
[c]
struct _zend_executor_globals {
zval **return_value_ptr_ptr;
zval uninitialized_zval;
zval *uninitialized_zval_ptr;
zval error_zval;
zval *error_zval_ptr;
zend_ptr_stack arg_types_stack;
/* symbol table cache */
HashTable *symtable_cache[SYMTABLE_CACHE_SIZE];
HashTable **symtable_cache_limit;
HashTable **symtable_cache_ptr;
zend_op **opline_ptr;
HashTable *active_symbol_table; //局部变量
HashTable symbol_table; /* main symbol table */ //全局变量
HashTable included_files; /* files already included */ //include的文件
JMP_BUF *bailout;
int error_reporting;
int orig_error_reporting;
int exit_status;
zend_op_array *active_op_array;
HashTable *function_table; /* function symbol table */ //函数表
HashTable *class_table; /* class table */ //类表
HashTable *zend_constants; /* constants table */ //常量表
zend_class_entry *scope;
zend_class_entry *called_scope; /* Scope of the calling class */
zval *This;
long precision;
int ticks_count;
zend_bool in_execution;
HashTable *in_autoload;
zend_function *autoload_func;
zend_bool full_tables_cleanup;
/* for extended information support */
zend_bool no_extensions;
#ifdef ZEND_WIN32
zend_bool timed_out;
OSVERSIONINFOEX windows_version_info;
#endif
HashTable regular_list;
HashTable persistent_list;
zend_vm_stack argument_stack;
int user_error_handler_error_reporting;
zval *user_error_handler;
zval *user_exception_handler;
zend_stack user_error_handlers_error_reporting;
zend_ptr_stack user_error_handlers;
zend_ptr_stack user_exception_handlers;
zend_error_handling_t error_handling;
zend_class_entry *exception_class;
/* timeout support */
int timeout_seconds;
int lambda_count;
HashTable *ini_directives;
HashTable *modified_ini_directives;
zend_objects_store objects_store;
zval *exception, *prev_exception;
zend_op *opline_before_exception;
zend_op exception_op[3];
struct _zend_execute_data *current_execute_data;
struct _zend_module_entry *current_module;
zend_property_info std_property_info;
zend_bool active;
void *saved_fpu_cw;
void *reserved[ZEND_MAX_RESERVED_RESOURCES];
};
[/c]
这里先简单看看,以后用到的时候再细说,
-
PHP里最基本的单元 变量:
在PHP里 定义一个变量 再简单不过了
如
[php]
[/php]
但是在内核中 它是用一个 zval结构体实现的
如上面定义变量 在内核中则执行了下面这些代码
[c]
zval *val;
MAKE_STD_ZVAL(val); //申请一块内存
ZVAL_STRING(val,"hello",1);//用ZVAL_STRING设置它的值为 "hello"
ZEND_SET_SYMBOL(EG(active_symbol_table),"a",val));//将 val指针加入到符号表里面去
[/c]
宏 MAKE_STD_ZVAL 定义如下
[c]
#define MAKE_STD_ZVAL(zv) \
ALLOC_ZVAL(zv); \ //它归根到底等于 (p) = (type *) emalloc(sizeof(type))
INIT_PZVAL(zv);
[/c]
INIT_PZVAL定义在
[c]
#define INIT_PZVAL(z) \ 看得出它是初始化参数
(z)->refcount__gc = 1; \
(z)->is_ref__gc = 0;
[/c]
那么 zval到底是什么呢
在zend/zend.h里面
typedef struct _zval_struct zval; //原来它是 _zval_struct 的别名
_zval_struct 定义如下
[c]
typedef union _zvalue_value {
long lval; //保存long类型的数据
double dval; //保存 double类型的数据
struct {
char *val; //真正的值在这里
int len; //这里返回长度
} str;
HashTable *ht;
zend_object_value obj; //这是一个对象
} zvalue_value;
struct _zval_struct {
zvalue_value value; //保存的值
zend_uint refcount__gc;//被引用的次数 如果为1 则只被自己使用如果大于1 则被其他变量以&的形式引用.
zend_uchar type; //数据类型 这也是 为什么 PHP是弱类型的原因
zend_uchar is_ref__gc; //表示是否为引用
};
[/c]
如果还是不够清楚..那么我们实战一下..用C来创建一个PHP变量
这里需要一个扩展,PHP如果用C扩展模块 这里就不说了
关键代码
[c]
PHP_FUNCTION(test_siren){
zval *value;
char *s="create a php variable";
value=(zval*)malloc(sizeof(zval));
memset(value,0,sizeof(value));
value->is_ref__gc=0; //非引用变量
value->refcount__gc=1;//引用次数 只有自己
value->type=IS_STRING;//类型为字符串
value->value.str.val=s;//值
value->value.str.len=strlen(s);//长度
ZEND_SET_SYMBOL(EG(active_symbol_table),"a",value);
}
[/c]
第三行和第四行的作用 与MAKE_STD_ZVAL的作用相同,给value分配内存空间
第5-9行 的作用与ZVAL_STRING的作用相同,
最后一行 是将value创建一个 在PHP里叫$a的变量..并添加到局部Hash表里..
这样 在PHP里
[php]
[/php]
就会输出 "create a php variable"
OK,
大功告成
注意,我是为了让大家看到PHP内部创建变量的流程 才采用C的形式创建变量,
绝对不推荐大家这样做.
还是一定要用PHP内部的内存管理机制分配并处理内存,\.
相关推荐
1. **哈希表(Hashtables)**:PHP内核中广泛使用哈希表存储数组、对象和变量。这是一种高效的数据结构,通过键值对进行查找和存储。哈希表的实现涉及到哈希函数、冲突解决策略以及内存管理。 2. **SAPI(Server ...
在了解PHP内核和扩展开发之前,我们需要了解一些基础知识,包括PHP的数据结构、内存管理以及哈希表的实现。 首先,让我们来谈谈PHP中的zval结构。zval是PHP内部用于存储变量的所有信息的数据结构。每一个变量在PHP...
- PHP内核中的HASH表和变量的管理是高效PHP代码的基石。 - PHP中的json_encode函数用于将变量编码为JSON格式,有助于数据交换。 - PHP内核研究还包括对PHP内核中各种功能和机制的深入理解,比如变量存储、函数...
本文将深入探讨PHP变量的内部实现,即PHP内核中的变量处理机制,包括变量的命名规则、数据类型、zval结构、哈希表等关键知识点。 PHP变量命名规则相对简单,以美元符号($)开头,后面跟着字母或下划线,之后可以包含...
### PHP内核介绍及扩展开发指南 #### 一、基础知识 **1.1.1 PHP变量的存储** 在PHP中,所有的变量都是通过`zval`结构体来存储的。`zval`的设计非常灵活,可以用来存储各种类型的值,包括整数、浮点数、字符串、...
从实践出发,继弱类型变量原理探究后,王帅将继续带大家弄清PHP内核中的一些常用部分,本期则是内核利器哈希表与哈希碰撞攻击。在PHP的Zend Engine(下面简称ZE)中,有一个非常重要的数据结构——哈希表...
在PHP内核中,Zend虚拟机使用HashTable结构体来存储包括变量和函数在内的上下文信息。因此,哈希表的性能直接关系到PHP执行的效率。如果哈希表因碰撞而性能下降,那么整个PHP程序的性能都将受到影响。 总结来说,...
创建PHP 5扩展涉及到对PHP内核的深入理解和操作。在本篇文章中,我们将探讨如何创建一个PHP 5扩展,了解PHP的生命周期,以及如何在扩展中添加对象和迭代器。 #### PHP数据处理方式 在开始编写扩展之前,我们需要了解...
然而,在PHP 5.3版本之后,增加了php_ini_get_configuration_hash接口,使得任何PHP扩展都能够直接访问configuration_hash,从而让配置信息的管理和修改变得更加灵活和开放。 需要注意的是,php.ini文件支持以数组...
一、 基础知识 本章简要介绍一些Zend引擎的内部机制,这些知识和Extensions密切相关,同时也可以帮助我们写出更加高效的PHP代码。 1.1 PHP变量的存储 1.1.1 zval结构 Zend使用zval结构来存储PHP变量的值,该结构如下...
常量与变量的主要区别在于它们的不可变性,这使得常量在定义了之后就不能再被修改。在扩展开发中,我们可以利用宏方法来自定义各种类型的常量,如null、bool、long、double、string等,因为这些是内核支持的基本数据...
- **优化Linux内核参数**:为了提高性能和稳定性,可能需要调整一些内核参数,如TCP缓冲区大小、文件句柄数量等。 - **平滑变更Nginx配置**:在不中断服务的情况下更新配置文件。 - **编写定时切割Nginx日志的脚本**...
... ...PHP7在内核方法的实现上和之前的版本存在显著的区别,导致原有...这通常要求开发者具有较高水平的C语言编程能力和对PHP内核的深刻理解。如果在迁移过程中遇到具体问题,还需参考PHP7的官方文档和社区提供的解决方案。