PHP 自动加载机制(__autoload,命名空间)

include 和 require 是PHP中引入文件的两个基本方法。在小规模开发中直接使用 include 和 require 没有什么不妥,但在大型项目中会造成大量的 include 和 require 堆积。这样的代码既不优雅,执行效率也很低,而且维护起来也相当困难。

为了解决这个问题,部分框架会给出一个引入文件的配置清单,在对象初始化的时候把需要的文件引入。但这只是让代码变得更简洁了一些,引入的效果仍然是差强人意。PHP5 之后,随着 PHP 面向对象支持的完善,__autoload 函数才真正使得自动加载成为可能。

  • include 和 require 功能是一样的,它们的不同在于 include 出错时只会产生警告,而 require 会抛出错误终止脚本。
  • include_once 和 include 唯一的区别在于 include_once 会检查文件是否已经引入,如果是则不会重复引入。

自动加载

实现自动加载最简单的方式就是使用 __autoload 魔术方法。当需要使用的类没有被引入时,这个函数会在PHP报错前被触发,未定义的类名会被当作参数传入。至于函数具体的逻辑,这需要用户自己去实现。

首先创建一个 autoload.php 来做一个简单的测试:

<?php

// 类未定义时,系统自动调用
function __autoload($class)
{
    /* 具体处理逻辑 */
    echo $class;// 简单的输出未定义的类名
}

new Test();

输出结果如下:

通过这个简单的例子可以发现,在类的实例化过程中,系统所做的工作大致是这样的:

<?php
/**
 * 模拟系统实例化过程
 *
 * @param string $class
 * @return void
 */
function instance($class)
{
    // 如果类存在则返回其实例
    if (class_exists($class, false)) {
        return new $class();
    }
    
    // 查看 autoload 函数是否被用户定义
    if (function_exists('__autoload')) {
        // 最后一次引入的机会
        __autoload($class); 
    }

    // 再次检查类是否存在
    if (class_exists($class, false)) {
        return new $class();
    } else { 
        // 类不存在
        throw new Exception('Class Not Found');
    }
}

明白了 __autoload 函数的工作原理之后,那就让我们来用它去实现自动加载。

首先创建一个类文件(Test.php),代码如下:

<?php
class Test {
    // 对象实例化时输出当前类名
    function __construct()
    {
        echo __class__;
    }
}

接下来我们就要定义 __autoload 的具体逻辑,使它能够实现自动加载,再创建一个文件(autoload.php),代码如下:

<?php
function __autoload($class)
{
    // 根据类名确定文件名
    $file = $class . '.php';

    if (file_exists($file)) {
        include $file; // 引入PHP文件
    }
}

new Test();

执行后输出如下:


命名空间

PHP的命名空间直到 PHP 5.3 之后才支持。

命名空间简而言之就是一种标识,它的主要目的是解决命名冲突的问题

命名空间通过关键字 namespace 来声明。如果一个文件中包含命名空间,它必须在其它所有代码之前声明命名空间。

<?php
namespace beijing;

class qinghua
{
    function __construct()
    {
        echo '北京清华大学';
    }
}

 如果不指定空间,则默认为全局(\)。在当前命名空间没有声明的情况下,限定名称和完全限定名称是等价的。

<?php
new beijing\qinghua();  // 限定名称
new \beijing\qinghua(); // 完全限定名称

下面的例子展示了在命名空间下,使用限定类名和完全限定类名的区别。(完全限定类名 = 当前命名空间 + 限定类名)

<?php
namespace china;
new beijing\qinghua();  // china\beijing\qinghua
new \beijing\qinghua(); // beijing\qinghua

使用命名空间只是让类名有了前缀,不容易发生冲突,系统仍然不会进行自动导入。

如果不引入文件,系统会在抛出 "Class Not Found" 错误之前触发 __autoload 函数,并将限定类名传入作为参数。

所以下面的例子都是基于已经将相关文件手动引入的情况下实现的,否则系统会抛出 "Class 'china\beijing' not found"。

<?php
/* 导入命名空间 */
use china\beijing;
new beijing(); // china\beijing

/* 设置别名 */
use china\beijing as Capital;
new Capital(); // china\beijing

/* 任何情况 */
new \china\beijing();// china\beijing

spl_autoload

接下来让我们要在含有命名空间的情况下去实现自动加载。这里我们使用 spl_autoload_register() 函数来实现,这需要你的 PHP 版本号大于 5.12。

spl_autoload_register 函数的功能就是把传入的函数(参数可以为回调函数或函数名称形式)注册到 SPL __autoload 函数队列中,并移除系统默认的 __autoload() 函数。

一旦调用 spl_autoload_register() 函数,当调用未定义类时,系统就会按顺序调用注册到 spl_autoload_register() 函数的所有函数,而不是自动调用 __autoload() 函数。

现在,我们来创建一个 beijing 类,它使用 china 作为它的命名空间(建议文件名与类名保持一致china.php):

<?php
namespace china; // 命名空间

class beijing // 类名
{
    function __construct()
    {
        echo __class__;
    }
}

接着,在同一个目录下新建一个 PHP 文件(test.php),使用 spl_autoload_register 以函数回调的方式实现自动加载: 

<?php
spl_autoload_register(function ($class) { // class = china\beijing

    /* 限定类名路径映射 */
    $class_map = array(
        // 限定类名 => 文件路径
        'china\\beijing' => './china.php',
    );

    /* 根据类名确定文件名 */
    $file = $class_map[$class];

    /* 引入相关文件 */
    if (file_exists($file)) {
        include $file;
    }
});

new \china\beijing();

执行test.php

这里我们使用了一个数组去保存类名与文件路径的关系,这样当类名传入时,自动加载器就知道该引入哪个文件去加载这个类了。

但是一旦文件多起来的话,映射数组会变得很长,这样的话维护起来会相当麻烦。如果命名能遵守统一的约定,就可以让自动加载器自动解析判断类文件所在的路径。

自动加载规范
1.PSR-0:https://github.com/PizzaLiu/P...
2.PSR-4:https://github.com/PizzaLiu/P...
3.PEAR:全是以"_"作为分隔

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值