함수를 호출하기 위해서는 함수 이름, 매개변수, 함수 정의(즉, 함수의 구체적인 실행 내용) 등 몇 가지 기본 정보가 필요합니다. 함수가 정의되어 있으면 이 함수의 이름이 무엇인지, 호출 시 어떤 매개변수가 전달되는지, 함수의 작업 내용은 무엇인지 자연스럽게 알게 됩니다. 그러나 Zend 엔진은 우리처럼 PHP 소스 코드를 "이해"할 수 없습니다. 코드를 실행하기 전에 먼저 처리해야 합니다. 우리는 여전히 다음 두 가지 작은 예부터 시작합니다.
<?php function foo(){ echo "I'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은 함수가 내부인지 여부를 결정하는 데 사용됩니다.
함수 실행
주의깊은 독자라면 함수가 호출될 때와 위 opcode의 함수 정의에 "함수 이름:"이 있다는 것을 알아차릴 수 있습니다. 실제로 사용자 정의 함수
内部函数的执行与用户函数不同。用户函数是php语句一条条“翻译”成op_line组成的一个op_array,而内部函数则是用C来实现的,因为执行环境也是C环境, 所以可以直接调用。如下面的例子:
<?php $foo = 'test'; 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函数为例。在<<第一节 函数的内部结构>>中, 我们知道内部函数所在的结构体中 有一个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 중국어 웹사이트의 기타 관련 기사를 참조하세요!