很重要:函数的实现
函数间的转换
在函数调用的执行代码中我们会看到这样一些强制转换:
EX(function_state).function = (zend_function *) op_array;
或者:
EG(active_op_array)
首先我们来看zend_function的结构,在Zend/zend_compile.h文件中,其定义如下:
typedef union _zend_function {
zend_uchar type;// 必须是第一个元素,在这个结构体中
struct{
zend_uchar type;//从来都不用
char *function_name;
zend_uint fn_flags;
union _zend_function *prototype;
zend_uint num_args;
zend_uint required_num_args;
zend_arg_info *arg_info;
zend_bool pass_rest_by_reference;
unsigned char return_reference;
}common;
zend_op_array op_array;
zend_internal_function internal_function;
}zend_function;
这是一个联合体,我们来温习一下联合体的一些特征。联合体的所有成员变量共享内存中的一块内存,在某个时刻只能有一个成员使用这块内存,并且当使用某一个成员时,其仅能按照她的类型和内存大小修改对应的内存空间。
#include <stdio.h>
#include <stdlib.h>
int main(){
typedef union _utype
{
int i;
char ch[2];
} utype;
utype a;
a.i = 10;
a.ch[0] = '1';
a.ch[1] = '1';
printf("a.i = %d a.ch = %s",a.i,a.ch);
getchar();
return (EXIT_SUCCESS);
}
程序输出:a.i = 12593 a.ch=11 当修改ch的值时,它会依据自己的规则覆盖i字段对应的内存空间。’1’对应的ASCII码值是49,二进制为00110001,当ch字段的两个元素都为’1’时,此时内存中存储的二进制为 00110001 00110001 转成十进制,其值为12593.
总结:php可以实现函数转换的关键点就是:
zend_internal_function,zend_function,zend_op_array这三种结构在一定程序上存在公共的元素,于是这些元素以联合体的形式共享内存,并且在执行过程中对于一个函数,这三种结构对应的字段在值上都是一样的,于是可以在一些结构间发生完美的强制类型转换。可以转换的列表如下:
zend_function可以与zend_op_array互换
zend_function可以与zend_internal_function互换。
其实zend_function 就是一个混合的数据结构,这种结构在一定程序上节省了内存空间。