php的函数包括用户定义的函数、内部函数(print_r count...)、匿名函数、变量函数($func = 'print_r'; $func(array('a','b'));)

PHP内核源码中将函数分为以下类型

#define ZEND_INTERNAL_FUNCTION              1
#define ZEND_USER_FUNCTION                  2
#define ZEND_OVERLOADED_FUNCTION            3
#define ZEND_EVAL_CODE                      4
#define ZEND_OVERLOADED_FUNCTION_TEMPORARY  5

一、用户函数(ZEND_USER_FUNCTION)

  函数不一定显式的有返回值,在PHP的实现中即使没有显式的返回,PHP内核也会帮我们返回NULL。

  ZEND在执行过程中,会将运行时信息存储于_zend_execute_data中:

struct _zend_execute_data {//...省略部分代码
    zend_function_state function_state;zend_function *fbc; /* Function Being Called *///...省略部分代码
};

  在程序初始化的过程中,function_state也会进行初始化,function_state由两个部分组成:

typedef struct _zend_function_state {zend_function *function;void **arguments;
} zend_function_state;

  *arguments是一个指向函数参数的指针,而函数体本事存储于*function中,*function是一个zend_function结构体,它最终存储了用户自定义函数的一切信息,具体结构如下:

typedef union _zend_function {zend_uchar type;    /* MUST be the first element of this struct! */struct {zend_uchar type;  /* never used */char *function_name;    //函数名称zend_class_entry *scope; //函数所在的类作用域zend_uint fn_flags;     //函数类型,如用户自定义则为 #define
ZEND_USER_FUNCTION 2  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;

  zend_function的结构体中的op_array存储了该函数中的所有操作,当函数被调用时,ZEND就会将这个op_array中的opline一条条顺序执行,并将最后的结果返回。函数的定义和执行是分开的,一个函数可以作为一个独立的运行单元存在。

二、内部函数(ZEND_INTERNAL_FUNCTION)

  ZEND_INTERNAL_FUNCTION函数是由扩展或者Zend/PHP内核提供的,用c/c++编写,可以直接执行的函数,以下为内部函数的结构

typedef struct _zend_internal_function {/* Common elements */zend_uchar type;char * function_name;zend_class_entry *scope;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;/* END of common elements */void (*handler)(INTERNAL_FUNCTION_PARAMETERS);struct _zend_module_entry *module;
} zend_internal_function;

  在模块初始化的时候,ZE会遍历每个载入的扩展模块,然后将模块中function_entry中指明的每一个函数(module->functions),创建一个zend_internal_function结构,并将其type设置为ZEND_INTERNAL_FUNCTION,将这个结构填入全局的函数表(HashTable结构);函数设置及注册过程见Zend/zene_API.c文件中的zend_register_function函数,这个函数除了处理函数页也处理类的方法,包括那些魔术方法。

  内部函数的结构与用户自定义函数结构基本类似,有一些不同:

  •   调用方法,handler字段,如果是ZEND_INTERNAL_FUNCTION,那么ZEND就会调用zend_execute_internal,通过zend_internal_function.handler来执行这个函数。而用户自定义函数需要生成中间代码,然后通过中间代码映射到相对就把方法调用。
  • 内置函数在结构中多了一个module字段,表示属于哪个模块。不同的扩展模块不同
  • type字段,在用户自定义函数中,type字段几乎无用,而内置函数中的type字段作为几种内部函数的区分。

三、变量函数

  如果一个变量名后边有圆括号,php将寻找与变量的值同名的函数,并且尝试执行。

  变量函数$func

$func = 'print_r';
$func('i am print_r function.');

  编译后中间代码

function name:  (null)
number of ops:  9
compiled vars:  !0 = $func
line     # *  op                           fetch          ext  return operands
------------------------------------------------------------------------------
-
-2     0  >   EXT_STMT1      ASSIGN                                                   !0,
'print_r'3     2      EXT_STMT3      INIT_FCALL_BY_NAME                                       !04      EXT_FCALL_BEGIN5      SEND_VAL
'i+am+print_r+function.'6      DO_FCALL_BY_NAME                              17      EXT_FCALL_END8    > RETURN                                  1

  内部函数

print_r('i am print_r function.');

  编译后中间代码

function name:  (null)
number of ops:  6
compiled vars:  none
line     # *  op                           fetch          ext  return  operands
-------------------------------------------------------------------------------
-
-2     0  >   EXT_STMT1      EXT_FCALL_BEGIN2      SEND_VAL
'i+am+print_r+function.'3      DO_FCALL                                      1
'print_r'4      EXT_FCALL_END5    > RETURN                                                   1

  对比发现,二者在调用中间代码上存在一些区别,变量函数是DO_FCALL_BY_NAME,而内部函数是DO_FCALL。这在语法解析时就已经决定了,见Zend/zend_complie.c文件的zend_do_end_function_call函数中部分代码:

if (!is_method && !is_dynamic_fcall && function_name->op_type==IS_CONST) {opline->opcode = ZEND_DO_FCALL;opline->op1 = *function_name;ZVAL_LONG(&opline->op2.u.constant,
zend_hash_func(Z_STRVAL(function_name->u.constant), Z_STRLEN(function_name-
>u.constant) + 1));} else {opline->opcode = ZEND_DO_FCALL_BY_NAME;SET_UNUSED(opline->op1);}

  如果不是方法,并且不是动态调用,并且函数名为字符串变量,则其生成的中间代码为ZEND_DO_FCALL。其他情况则为ZEND_DO_FCALL_BY_NAME。另外将变量函数作为回调函数,其处理过程在Zend/zend_complie.c文件的zend_do_pass_param函数中,最终会体现在中间代码执行过程中的ZEND_SEND_VAL_SPEC_CONST_HADNLER等函数中。

四、匿名函数

  匿名函数是一类不需要指定表示符,而又可以被调用的函数或子例程,匿名函数可以方便的作为参数传递给其他函数。

  

深入理解PHP内核(五)函数的内部结构相关推荐

  1. php内核函数手册,深入理解PHP内核(五)函数的内部结构,深入理解内部结构

    深入理解PHP内核(五)函数的内部结构,深入理解内部结构 php的函数包括用户定义的函数.内部函数(print_r count...).匿名函数.变量函数($func = 'print_r'; $fu ...

  2. tipi 深入理解php内核 pdf_大牛的学习笔记-深入理解Linux内核(完整版)

    第一章.绪论 1.Unix文件可以是下列类型之一: a.正规文件(regular file) b.目录(directroy) c.符号链(symbolic link) d.块设备文件(block-or ...

  3. 深入理解Linux内核-第3版 译者序、前言、目录 内核2.6.11

    一.译者序 Linux是一个全新的世界,世界意味着博大精深,而新或许代表对旧的割舍和扬弃,加在一起,就是要我们在割舍和扬弃的同时还要积累知识到博大精深的地步,这容易做到吗?是的,这不容易做到.Gera ...

  4. 深入理解 Linux 内核

    Linux 内核系列文章 Linux 内核设计与实现 深入理解 Linux 内核 深入理解 Linux 内核(二) Linux 设备驱动程序 Linux设备驱动开发详解 文章目录 Linux 内核系列 ...

  5. 笔记:深入理解Linux内核(二)

    笔记:深入理解Linux内核(二) 二零二一年十月二十四日 文章目录 笔记:深入理解Linux内核(二) 第二章:内存寻址 内存地址 硬件中的分段 段选择符和段选择器 段描述符 快速访问段描述符 分段 ...

  6. 深入理解Linux内核(一)——Linux操作系统基础概念

    文章目录 前言 操作系统基本概念 多用户系统 用户和组 进程 内核体系结构 Unix文件系统概述 文件 硬链接和软链接 文件类型 文件描述符与索引节点 访问权限和文件模式 文件操作的系统调用 打开文件 ...

  7. python函数是一段具有特定功能的语句组_Python学习笔记(五)函数和代码复用

    本文将为您描述Python学习笔记(五)函数和代码复用,具体完成步骤: 函数能提高应用的模块性,和代码的重复利用率.在很多高级语言中,都可以使用函数实现多种功能.在之前的学习中,相信你已经知道Pyth ...

  8. TLD(Tracking-Learning-Detection)学习与源码理解之(五)

    TLD(Tracking-Learning-Detection)学习与源码理解之(五)   zouxy09@qq.com http://blog.csdn.net/zouxy09 下面是自己在看论文和 ...

  9. 深入理解Linux内核之内存寻址

    说明: 本文基于第三版<深入理解 Linux 内核>,该部分以 80x86 处理器为基准进行介绍,并且略过了原文中详细介绍32位扩展分页部分. https://xcraft.tech/20 ...

最新文章

  1. linux ubuntu安装 mono,在Ubuntu 18.04系统中安装Mono及基本使用Mono的方法
  2. Linux系统中的软件管理
  3. java创建线程代码_Java创建与结束线程代码示例
  4. Php基础数学运算篇
  5. Linux下配置安装PHP环境
  6. 20岁没有学历能学计算机,20岁没学历学什么技术怎么办?学什么比较吃香?
  7. Delphi XE7 用indy开发微信公众平台(3)- 验证消息真实性
  8. Akka-Cluster(0)- 分布式应用开发的一些想法
  9. iOS媒体视频播放器应用源码
  10. 绿油损耗大于大多数高速板材,对于高度板材而言,绿油带来的损耗会更明显
  11. HashMap 底层
  12. VUE中常用的UI库
  13. 一文讲解AGV机器人的12种导航导引方式,收藏备用
  14. MyBatis配置文件
  15. 20220505模拟赛总结(ABC237)
  16. 如何成为一名数据科学家?
  17. Django admin 页面添加自定义按钮 点击事件
  18. PostgreSQL 多维空间几何对象 相交、包含 高效率检索实践 - cube
  19. BTC呈现跳水行情或与全球石油价格暴跌有关
  20. Windows桌面图标不见了,可能是结束了explorer.exe进程导致

热门文章

  1. php table字段排序,jQuery如何对table进行排序操作的示例详解
  2. 如何拼通网络ip地址_如何解决IP地址冲突
  3. java bufferedwriter 写入tab_一个Java程序员的成长历程(014,015)天
  4. Linux编译lclntsh,Linux下编译C/C++时,出现/usr/bin/ld: cannot find -l*** 错误的处理
  5. XML 解析XML文档 XML约束
  6. ==和equals()的作用及区别
  7. 服务器日志显示乱码,CentosOS 6.5 服务器 控制台输出中文乱码,日志打印中文也乱码...
  8. java获取date的时分秒_Java 之 Date 获取 年月日时分秒
  9. java file_Java IO: File
  10. stata命令汇总_第九届高级计量经济学及stata应用研讨会在京顺利举办