### 3.4.6 類的自動加載
在實際使用中,通常會把一個類定義在一個文件中,然后使用時include加載進來,這樣就帶來一個問題:在每個文件的頭部都需要包含一個長長的include列表,而且當文件名稱修改時也需要把每個引用的地方都改一遍,另外前面我們也介紹過,原則上父類需要在子類定義之前定義,當存在大量類時很難得到保證,因此PHP提供了一種類的自動加載機制,當使用未被定義的類時自動調用類加載器將類加載進來,方便類的同一管理。
在內核實現上類的自動加載實際就是定義了一個鉤子函數,實例化類時如果在EG(class_table)中沒有找到對應的類則會調用這個鉤子函數,調用完以后再重新查找一次。這個鉤子函數保存在EG(autoload_func)中。
PHP中提供了兩種方式實現自動加載:`__autoload()`、`spl_autoload_register()`。
***(1)__autoload():***
這種方式比較簡單,用戶自定義一個`__autoload()`函數即可,參數是類名,當實例化一個類是如果沒有找到這個類則會查找用戶是否定義了`__autoload()`函數,如果定義了則調用此函數,比如:
```php
//文件1:my_class.php
<?php
class my_class {
public $id = 123;
}
//文件2:b.php
<?php
function __autoload($class_name){
//do something...
include $class_name . '.php';
}
$obj = new my_class();
var_dump($obj);
```
__(2)spl_autoload_register():__
相比`__autoload()`只能定義一個加載器,`spl_autoload_register()`提供了更加靈活的注冊方式,可以支持任意數量的加載器,比如第三方庫加載規則不可能保持一致,這樣就可以通過此函數注冊自己的加載器了,在實現上spl創建了一個隊列來保存用戶注冊的加載器,然后定義了一個spl_autoload函數到EG(autoload_func),當找不到類時內核回調spl_autoload,這個函數再依次調用用戶注冊的加載器,沒調用一個重新檢查下查找的類是否在EG(class_table)中已經注冊,仍找不到的話繼續調用下一個加載器,直到類成功注冊為止。
```c
bool spl_autoload_register ([ callable $autoload_function [, bool $throw = true [, bool $prepend = false ]]] )
```
參數`$autoload_function`為加載器,可以是函數名,第2個參數`$throw`用于設置autoload_function 無法成功注冊時, spl_autoload_register()是否拋出異常,最后一個參數如果為true時spl_autoload_register() 會添加函數到隊列之首,而不是隊列尾部。
```php
function autoload_one($class_name){
echo "autoload_one->", $class_name, "\n";
}
function autoload_two($class_name){
echo "autoload_two->", $class_name, "\n";
}
spl_autoload_register("autoload_one");
spl_autoload_register("autoload_two");
$obj = new my_class();
var_dump($obj);
```
這個例子執行時就會將autoload_one()、autoload_two()都調一遍,假如第一個函數就成功注冊了my_class類則不會再調后面的加載器。
內核查找類通過`zend_lookup_class_ex()`完成,我們簡單看下其處理過程。
```c
//file: zend_execute_API.c
ZEND_API zend_class_entry *zend_lookup_class_ex(zend_string *name, const zval *key, int use_autoload)
{
...
//從EG(class_table)符號表找類的zend_class_entry,如果找到說明類已經編譯,直接返回
ce = zend_hash_find_ptr(EG(class_table), lc_name);
if (ce) {
if (!key) {
zend_string_release(lc_name);
}
return ce;
}
...
//如果沒有通過spl注冊則看下是否定義了__autoload()
if (!EG(autoload_func)) {
zend_function *func = zend_hash_str_find_ptr(EG(function_table), "__autoload", sizeof("__autoload") - 1);
if (func) {
EG(autoload_func) = func;
} else {
return NULL;
}
}
...
fcall_cache.function_handler = EG(autoload_func);
...
//調用EG(autoload_func)函數,然后再查一次EG(class_table)
if ((zend_call_function(&fcall_info, &fcall_cache) == SUCCESS) && !EG(exception)) {
ce = zend_hash_find_ptr(EG(class_table), lc_name);
}
...
}
```
SPL的具體實現比較簡單,這里不再介紹。
- 前言
- 第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超時控制的思考