### 3.2.2 內部函數
上一節已經提過,內部函數指的是由內核、擴展提供的C語言編寫的function,這類函數不需要經歷opcode的編譯過程,所以效率上要高于PHP用戶自定義的函數,調用時與普通的C程序沒有差異。
Zend引擎中定義了很多內部函數供用戶在PHP中使用,比如:define、defined、strlen、method_exists、class_exists、function_exists......等等,除了Zend引擎中定義的內部函數,PHP擴展中也提供了大量內部函數,我們也可以靈活的通過擴展自行定制。
#### 3.2.2.1 內部函數結構
上一節介紹`zend_function`為union,其中`internal_function`就是內部函數用到的,具體結構:
```c
//zend_complie.h
typedef struct _zend_internal_function {
/* 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_internal_arg_info *arg_info;
/* END of common elements */
void (*handler)(INTERNAL_FUNCTION_PARAMETERS); //函數指針,展開:void (*handler)(zend_execute_data *execute_data, zval *return_value)
struct _zend_module_entry *module;
void *reserved[ZEND_MAX_RESERVED_RESOURCES];
} zend_internal_function;
```
`zend_internal_function`頭部是一個與`zend_op_array`完全相同的common結構。
下面看下如何定義一個內部函數。
#### 3.2.2.2 定義與注冊
內部函數與用戶自定義函數沖突,用戶無法在PHP代碼中覆蓋內部函數,執行PHP腳本時會提示error錯誤。
內部函數的定義非常簡單,我們只需要創建一個普通的C函數,然后創建一個`zend_internal_function`結構添加到 __EG(function_table)__ (也可能是CG(function_table),取決于在哪一階段注冊)中即可使用,內部函數 __通常__ 情況下是在php_module_startup階段注冊的,這里之所以說通常是按照標準的擴展定義,除了擴展提供的方式我們可以在任何階段自由定義內部函數,當然并不建議這樣做。下面我們先不討論擴展標準的定義方式,我們先自己嘗試下如何注冊一個內部函數。
根據`zend_internal_function`的結構我們知道需要定義一個handler:
```c
void qp_test(INTERNAL_FUNCTION_PARAMETERS)
{
printf("call internal function 'qp_test'\n");
}
```
然后創建一個內部函數結構(我們在擴展PHP_MINIT_FUNCTION方法中注冊,也可以在其他位置):
```c
PHP_MINIT_FUNCTION(xxxxxx)
{
zend_string *lowercase_name;
zend_function *reg_function;
//函數名轉小寫,因為php的函數不區分大小寫
lowercase_name = zend_string_alloc(7, 1);
zend_str_tolower_copy(ZSTR_VAL(lowercase_name), "qp_test", 7);
lowercase_name = zend_new_interned_string(lowercase_name);
reg_function = malloc(sizeof(zend_internal_function));
reg_function->internal_function.type = ZEND_INTERNAL_FUNCTION; //定義類型為內部函數
reg_function->internal_function.function_name = lowercase_name;
reg_function->internal_function.handler = qp_test;
zend_hash_add_ptr(CG(function_table), lowercase_name, reg_function); //注冊到CG(function_table)符號表中
}
```
接著編譯、安裝擴展,測試:
```php
qp_test();
```
結果輸出:
`call internal function 'qp_test'`
這樣一個內部函數就定義完成了。這里有一個地方需要注意的我們把這個函數注冊到 __CG(function_table)__ 中去了,而不是 __EG(function_table)__ ,這是因為在`php_request_startup`階段會把 __CG(function_table)__ 賦值給 __EG(function_table)__ 。
上面的過程看著比較簡單,但是在實際應用中不要這樣做,PHP提供給我們一套標準的定義方式,接下來看下如何在擴展中按照官方方式提供一個內部函數。
首先也是定義C函數,這個通過`PHP_FUNCTION`宏定義:
```c
PHP_FUNCTION(qp_test)
{
printf("call internal function 'qp_test'\n");
}
```
然后是注冊過程,這個只需要我們將所有的函數數組添加到擴展結構`zend_module_entry.functions`即可,擴展加載過程中會自動進行函數注冊(見1.2節),不需要我們干預:
```c
const zend_function_entry xxxx_functions[] = {
PHP_FE(qp_test, NULL)
PHP_FE_END
};
zend_module_entry xxxx_module_entry = {
STANDARD_MODULE_HEADER,
"擴展名稱",
xxxx_functions,
PHP_MINIT(timeout),
PHP_MSHUTDOWN(timeout),
PHP_RINIT(timeout), /* Replace with NULL if there's nothing to do at request start */
PHP_RSHUTDOWN(timeout), /* Replace with NULL if there's nothing to do at request end */
PHP_MINFO(timeout),
PHP_TIMEOUT_VERSION,
STANDARD_MODULE_PROPERTIES
};
```
關于更多擴展中函數相關的用法會在后面擴展開發一章中詳細介紹,這里不再展開。
- 前言
- 第1章 PHP基本架構
- 1.1 PHP簡介
- 1.2 PHP7的改進
- 1.3 FPM
- 1.3.1 概述
- 1.3.2 基本實現
- 1.3.3 FPM的初始化
- 1.3.4 請求處理
- 1.3.5 進程管理
- 1.4 PHP執行的幾個階段
- 第2章 變量
- 2.1 變量的內部實現
- 2.2 數組
- 2.3 靜態變量
- 2.4 全局變量
- 2.5 常量
- 第3章 Zend虛擬機
- 3.1 PHP代碼的編譯
- 3.1.1 詞法解析、語法解析
- 3.1.2 抽象語法樹編譯流程
- 3.2 函數實現
- 3.2.1 內部函數
- 3.2.2 用戶函數的實現
- 3.3 Zend引擎執行流程
- 3.3.1 基本結構
- 3.3.2 執行流程
- 3.3.3 函數的執行流程
- 3.3.4 全局execute_data和opline
- 3.4 面向對象實現
- 3.4.1 類
- 3.4.2 對象
- 3.4.3 繼承
- 3.4.4 動態屬性
- 3.4.5 魔術方法
- 3.4.6 類的自動加載
- 3.5 運行時緩存
- 3.6 Opcache
- 3.6.1 opcode緩存
- 3.6.2 opcode優化
- 3.6.3 JIT
- 第4章 PHP基礎語法實現
- 4.1 類型轉換
- 4.2 選擇結構
- 4.3 循環結構
- 4.4 中斷及跳轉
- 4.5 include/require
- 4.6 異常處理
- 第5章 內存管理
- 5.1 Zend內存池
- 5.2 垃圾回收
- 第6章 線程安全
- 6.1 什么是線程安全
- 6.2 線程安全資源管理器
- 第7章 擴展開發
- 7.1 概述
- 7.2 擴展的實現原理
- 7.3 擴展的構成及編譯
- 7.3.1 擴展的構成
- 7.3.2 編譯工具
- 7.3.3 編寫擴展的基本步驟
- 7.3.4 config.m4
- 7.4 鉤子函數
- 7.5 運行時配置
- 7.5.1 全局變量
- 7.5.2 ini配置
- 7.6 函數
- 7.6.1 內部函數注冊
- 7.6.2 函數參數解析
- 7.6.3 引用傳參
- 7.6.4 函數返回值
- 7.6.5 函數調用
- 7.7 zval的操作
- 7.7.1 新生成各類型zval
- 7.7.2 獲取zval的值及類型
- 7.7.3 類型轉換
- 7.7.4 引用計數
- 7.7.5 字符串操作
- 7.7.6 數組操作
- 7.8 常量
- 7.9 面向對象
- 7.9.1 內部類注冊
- 7.9.2 定義成員屬性
- 7.9.3 定義成員方法
- 7.9.4 定義常量
- 7.9.5 類的實例化
- 7.10 資源類型
- 7.11 經典擴展解析
- 7.8.1 Yaf
- 7.8.2 Redis
- 第8章 命名空間
- 8.1 概述
- 8.2 命名空間的定義
- 8.2.1 定義語法
- 8.2.2 內部實現
- 8.3 命名空間的使用
- 8.3.1 基本用法
- 8.3.2 use導入
- 8.3.3 動態用法
- 附錄
- break/continue按標簽中斷語法實現
- defer推遲函數調用語法的實現
- 一起線上事故引發的對PHP超時控制的思考