首先认识一个函数:
spl_autoload_register
官方文档:https://www.php.net/manual/zh/function.spl-autoload-register.php
spl_autoload_register() 满足了多条 autoload 的需求。 它实际上创建了 autoload 函数的队列,按定义时的顺序逐个执行。相比之下, __autoload() 只可以定义一次。
魔术方法__autoload
没啥好说的,上段代码:
function __autoload($class)
{
// 根据类名确定文件名
$file = $class . '.php';
if (file_exists($file)) {
include $file; // 引入PHP文件
}
}
spl_autoload_register
我们来模拟一次自动加载:
先建立一个项目,文件夹结构如下:
先写一个自动加载封装类:Loader.php
<?php
class Loader
{
// 解析命名空间的路径
public static $vendorMap = [
'app' => __DIR__ . DIRECTORY_SEPARATOR . 'app',
];
public function __construct()
{
echo __DIR__;die;
}
// 自动加载器
public static function autoload($class)
{
$file = self::findFile($class);
if (file_exists($file)) {
self::includeFile($file);
}
}
// 解析文件路径
private static function findFile($class)
{
$vendor = substr($class, 0, strpos($class, '\\')); // 顶级命名空间
$vendorDir = self::$vendorMap[$vendor]; // 文件基目录
$filePath = substr($class, strlen($vendor)) . '.php'; // 文件相对路径
return strtr($vendorDir . $filePath, '\\', DIRECTORY_SEPARATOR); // 文件标准路径
}
// 引入文件
private static function includeFile($file)
{
if (is_file($file)) {
include $file;
}
}
}
写一个简单的类文件index.php
<?php
namespace app\index\controller\hello;
class Index
{
function test()
{
echo '<h1>hello world</h1>';
}
}
我们加载一下封装好的自动加载类,模拟一下index类的自动加载:run.php
<?php
include 'Loader.php'; // 引入加载器
spl_autoload_register('Loader::autoload'); // 注册自动加载
$indexController = new app\index\controller\hello\Index(); // 实例化未引用的类
$indexController->test();
我们运行run.php,可以看到index.php可以直接实例化了,而我们的文件中并没有直接引入他,而只是注册了一个自动加载器,是不是相当方便呐!
就这,奥利给!