PHP는 실제로 Zend 커널의 관점에서 보면 일반적인 C 프로그램입니다. 우리가 작성하는 PHP 코드는 이 프로그램의 입력입니다. 커널 PHP 코드를 인식할 수 있는 C 프로그램으로 "변환"하는 프로세스가 PHP 컴파일입니다.
추천 과정: PHP Tutorial
C 프로그램은 컴파일 중에 코드 줄을 기계어로 컴파일하며 이러한 명령어는 실행 중에 컴파일된 바이너리 프로그램에 기록됩니다. 바이너리 프로그램을 해당 메모리 영역(상수 영역, 데이터 영역, 코드 영역)에 넣고 실행 중인 스택을 할당한 다음 코드 영역의 처음부터 실행을 시작하는 것은 C 프로그램을 컴파일하고 실행하는 간단한 과정입니다.
마찬가지로 PHP의 컴파일은 PHP 코드가 기계어 코드로 컴파일되지 않고 여러 opcode 배열로 구문 분석된다는 점을 제외하면 일반 C 프로그램의 컴파일과 유사합니다. 각 opcode는 C의 일반 구조체이며 그 의미는 해당됩니다. C 프로그램의 기계 명령어에 대한 실행 프로세스는 엔진이 순서대로 opcode를 실행하는 것입니다. 예를 들어 PHP에서 변수를 정의합니다. $a = 123;
커널의 최종 실행은 메모리 조각을 작성하는 것입니다. 그 안에 담긴 가치.
zend_compile.h 파일에서 opcode 구조는 다음과 같습니다.
struct _zend_op { const void *handler; //对应执行的C语言function,即每条opcode都有一个C function处理 znode_op op1; //操作数1 znode_op op2; //操作数2 znode_op result; //返回值 uint32_t extended_value; uint32_t lineno; zend_uchar opcode; //opcode指令 zend_uchar op1_type; //操作数1类型 zend_uchar op2_type; //操作数2类型 zend_uchar result_type; //返回值类型 };
따라서 PHP 구문 분석 프로세스의 작업은 PHP 코드(어휘 분석 re2c, 구문 분석 bison을 통해)를 opcode 배열로 변환하는 것입니다. 코드는 opcode에 저장된 다음 실행을 위해 opcode 배열을 zend 엔진에 전달합니다. Opcode는 할당, 덧셈 및 뺄셈 연산, 함수 호출 등과 같이 커널에 의해 특별히 실행되는 명령입니다. 각 opcode는 처리에 해당합니다. 핸들이며 이러한 핸들러는 미리 정의된 C 함수입니다.
struct _zend_op_array { //common是普通函数或类成员方法对应的opcodes快速访问时使用的字段 /* Common elements */ zend_uchar type; zend_uchar arg_flags[3]; /* bitset of arg_info.pass_by_reference */ uint32_t fn_flags; zend_string *function_name; zend_class_entry *scope; zend_function *prototype; uint32_t num_args; uint32_t required_num_args; zend_arg_info *arg_info; /* END of common elements */ uint32_t *refcount; uint32_t last; //opcode指令数组 zend_op *opcodes; //PHP代码里定义的变量数:op_type为IS_CV的变量,不含IS_TMP_VAR、IS_VAR的 //编译前此值为0,然后发现一个新变量这个值就加1 int last_var; //临时变量数:op_type为IS_TMP_VAR、IS_VAR的变量 uint32_t T; //PHP变量名数组 zend_string **vars;//这个数组在ast编译期间配合last_var用来确定各个变量的编号,非常重要的一步操作 int last_live_range; int last_try_catch; zend_live_range *live_range; zend_try_catch_element *try_catch_array; //静态变量符号表:通过static声明的 /* static variables support */ HashTable *static_variables; zend_string *filename; uint32_t line_start; uint32_t line_end; zend_string *doc_comment; uint32_t early_binding; /* the linked list of delayed declarations */ //字面量数量 int last_literal; //字面量(常量)数组,这些都是在PHP代码定义的一些值 zval *literals; //运行时缓存数组大小 int cache_size; //运行时缓存,主要用于缓存一些znode_op以便于快速获取数据,后面单独介绍这个机制 void **run_time_cache; void *reserved[ZEND_MAX_RESERVED_RESOURCES]; };
opcode 명령: 바이너리 프로그램의 코드 세그먼트에 해당하는 PHP 코드에 해당하는 특정 처리 작업
리터럴 저장소: 함수 이름, 클래스 이름, 상수라고 하는 PHP 코드에 정의된 변수의 일부 초기 값 리터럴이라고 불리는 이 값은 실행 중에 변수, 함수 호출 등을 초기화하는 데 사용됩니다. 변수 할당: 리터럴과 유사하며 각 변수에 의해 정의된 변수 및 임시 변수의 수를 나타냅니다. 해당 숫자가 있으면 초기화를 실행하고 총 개수에 따라 zval을 한 번에 할당합니다. 사용 시 변수 이름에 따라 인덱싱하는 대신 숫자에 따라 인덱싱됩니다. 어떻게 구현합니까?
위 내용은 PHP 파일을 컴파일하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!