1

初衷

有时候我们会遇到对php提供的函数执行结果感到诧异,这时候我们需要看下源码的实现,才能做出更好的解释。

代码结构

首先在php.net官网下载php源码。

下载好的源码,我们需要重点关注的是ext和zend两个目录。其他目录可以不用关心。

clipboard.png

ext目录下放的是php里面的扩展。包括我们经常使用的一些核心函数(json_encode、json_decode),同时也包括mysqli、PDO等核心类。

zend目录下放的是zend引擎的源码。控制PHP代码运行时候的运行环境。它处理PHP提供的所有“语言层”的特性,包括:变量,表达式,语法解析,代码执行和错误处理。

如何查找源码

假设我们想知道json_encode怎么实现,只需在整个目录中搜索 _function(json_encode)。就可以发现json_encode的源码是在json.c文件中的。

clipboard.png

同理找strlen方法。

clipboard.png

仔细看就会发现,两个搜索结果略有差异。 一个是PHP_FUNCTION,一个是ZEND_FUNCTION。一个是zend引擎定义的函数,一个是PHP扩展的函数。

在json.c文件中,可以很轻松找出json_encode的实现代码如下。

/* {{{ proto string json_encode(mixed data [, int options[, int depth]])
   Returns the JSON representation of a value */
static PHP_FUNCTION(json_encode)
{
    zval *parameter;
    smart_str buf = {0};
    zend_long options = 0;
    zend_long depth = PHP_JSON_PARSER_DEFAULT_DEPTH;

    if (zend_parse_parameters(ZEND_NUM_ARGS(), "z|ll", &parameter, &options, &depth) == FAILURE) {
        return;
    }

    JSON_G(error_code) = PHP_JSON_ERROR_NONE;

    JSON_G(encode_max_depth) = (int)depth;

    php_json_encode(&buf, parameter, (int)options);

    if (JSON_G(error_code) != PHP_JSON_ERROR_NONE && !(options & PHP_JSON_PARTIAL_OUTPUT_ON_ERROR)) {
        smart_str_free(&buf);
        ZVAL_FALSE(return_value);
    } else {
        smart_str_0(&buf); /* copy? */
        ZVAL_NEW_STR(return_value, buf.s);
    }
}
/* }}} */

剩下的就是要读懂这段代码,后面再讲……o(╯□╰)o


hyman66
10 声望3 粉丝