>  기사  >  백엔드 개발  >  PHP 파일을 컴파일하는 방법

PHP 파일을 컴파일하는 방법

(*-*)浩
(*-*)浩원래의
2019-05-11 16:52:558410검색

PHP는 실제로 Zend 커널의 관점에서 보면 일반적인 C 프로그램입니다. 우리가 작성하는 PHP 코드는 이 프로그램의 입력입니다. 커널 PHP 코드를 인식할 수 있는 C 프로그램으로 "변환"하는 프로세스가 PHP 컴파일입니다.

추천 과정: PHP Tutorial

PHP 파일을 컴파일하는 방법

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 컴파일 프로세스에는 re2c 및 bison을 사용하여 완성되는 어휘 분석이 포함됩니다. 이전 PHP 버전은 opcode를 직접 생성합니다. PHP7은 구문 분석 단계에서 새로운 추상 구문 트리(AST)를 생성합니다. opcode 배열이 생성됩니다

위 내용은 PHP 파일을 컴파일하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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