>  기사  >  백엔드 개발  >  PHP 사용자 정의 함수 호출 및 실행 과정에 대한 자세한 설명

PHP 사용자 정의 함수 호출 및 실행 과정에 대한 자세한 설명

伊谢尔伦
伊谢尔伦원래의
2017-06-27 09:37:393887검색

함수를 호출하기 위해서는 함수 이름, 매개변수, 함수 정의(즉, 함수의 구체적인 실행 내용) 등 몇 가지 기본 정보가 필요합니다. 함수가 정의되어 있으면 이 함수의 이름이 무엇인지, 호출 시 어떤 매개변수가 전달되는지, 함수의 작업 내용은 무엇인지 자연스럽게 알게 됩니다. 그러나 Zend 엔진은 우리처럼 PHP 소스 코드를 "이해"할 수 없습니다. 코드를 실행하기 전에 먼저 처리해야 합니다. 우리는 여전히 다음 두 가지 작은 예부터 시작합니다.

<?php
    function foo(){
        echo "I&#39;m foo!";
    }   
    foo();
?>

아래에서 해당 OPCODES를 살펴보겠습니다.

함수 이름: (null) line # * OP FETCH EXT Results

Operands


----- ------------------------------------- ------------------------------------- ------------------------------------- ------------------------------------- --------- --------------- ---------------------- -----------
                                                                            DO_FCALL              ~ ~                                                                                   내선 가져오기 피연산자 반환

-------------------------------------------------- --- ----------------------------------

4 0 > ECHO                                                                                              >                    ~로
5 1 >돌아가기 다음 섹션에서 자세히 설명하는 것처럼 구현이 특히 중요합니다. 이제 foo 함수 호출에 중점을 두겠습니다. foo를 호출하는 OPCODE는 "DO_FCALL"입니다. DO_FCALL이 함수 호출 연산을 수행할 때 ZE는 function_table에 있는 함수 이름에 따라 검색합니다(위에서 언급했듯이 여기서 함수 이름은 str_tolower에 의해 처리되므로 PHP의 함수 이름은 대소문자가 아닙니다. -sensitive) 함수의 정의입니다. 존재하지 않으면 "정의되지 않은 함수 xxx() 호출"이라는 오류 메시지가 보고됩니다. 존재하는 경우 함수의 zend_function 구조 포인터가 반환된 후 값이 반환됩니다. function.type은 함수가 내부인지 여부를 결정하는 데 사용됩니다.

함수는 여전히 사용자 정의 함수입니다. zend_execute_internal(zend_internal_function.handler)을 호출하거나 이 함수에 포함된 zend_op_array를 직접 실행하려면 zend_execute를 호출하세요.


함수 실행

주의깊은 독자라면 함수가 호출될 때와 위 opcode의 함수 정의에 "함수 이름:"이 있다는 것을 알아차릴 수 있습니다. 실제로 사용자 정의 함수

의 실행은 관련이 없습니다. 본질적으로 차이점은 함수 내의 PHP 문은 함수 외부의 PHP 문과 다르지 않다는 것입니다. 함수 본문 자체의 가장 큰 차이점은 실행 환경에 있습니다. 이 "실행 환경"의 가장 중요한 특징은 변수의 범위입니다. 우리 모두 알고 있듯이 함수 내에 정의된 변수는 함수 외부에서 직접 사용할 수 없으며 그 반대의 경우도 마찬가지입니다. 그리고 해당 기능이 실행되면 해당 기능에 진입하기 전의 환경정보를 반드시 저장해 놓아야 합니다. 기능이 실행된 후에는 이러한 환경 정보도 복원되어 전체 프로그램이 계속 실행될 수 있습니다.

内部函数的执行与用户函数不同。用户函数是php语句一条条“翻译”成op_line组成的一个op_array,而内部函数则是用C来实现的,因为执行环境也是C环境, 所以可以直接调用。如下面的例子:

<?php
    $foo = &#39;test&#39;;
    print_r($foo);
?>

生成的opcodes中,内部函数和用户函数的处理都是由DO_FCALL来进行的。而在其具体实现的zend_do_fcall_common_helper_SPEC()中, 则对是否为内部函数进行了判断,如果是内部函数,则使用一个比较长的调用

((zend_internal_function *) EX(function_state).function)->handler(opline->extended_value, EX_T(opline->result.u.var).var.ptr, EX(function_state).function->common      .return_reference?&EX_T(opline->result.u.var).var.ptr:NULL, EX(object), RETURN_VALUE_USED(opline) TSRMLS_CC);

上面这种方式的内部函数是在zend_execute_internal函数没有定义的情况下。而在而在Zend/zend.c文件的zend_startup函数中,

zend_execute_internal = NULL;

此函数确实被赋值为NULL。于是我们在if (!zend_execute_internal)判断时会成立,所以我们是执行那段很长的调用。 那么,这段很长的调用到底是什么呢?以我们常用的 count函数为例。在4a863a0d9769be17e9a49a855df6b6bb>中, 我们知道内部函数所在的结构体中 有一个handler指针指向此函数需要调用的内部定义的C函数。 这些内部函数在模块初始化时就以扩展的函数的形式加载到EG(function_table)。其调用顺序:

php_module_startup --> php_register_extensions --> zend_register_internal_module
--> zend_register_module_ex --> zend_register_functions
 
zend_register_functions(NULL, module->functions, NULL, module->type TSRMLS_CC)

在standard扩展中。module的定义为:

zend_module_entry basic_functions_module = {
    STANDARD_MODULE_HEADER_EX,
    NULL,
    standard_deps,
    "standard",                 /* extension name */
    basic_functions,            /* function list */
    ... //省略}

从上面的代码可以看出,module->functions是指向basic_functions。在basic_functions.c文件中查找basic_functions的定义。

위 내용은 PHP 사용자 정의 함수 호출 및 실행 과정에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.