欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

PHP自动加载机制实例详解

程序员文章站 2022-05-28 10:57:23
本文实例讲述了php自动加载机制。分享给大家供大家参考,具体如下: 在php中,我们一般使用 require, requre_once, include, include...

本文实例讲述了php自动加载机制。分享给大家供大家参考,具体如下:

在php中,我们一般使用 require, requre_once, include, include_once 这四个命令来加载其他php文件,这在一般小型的php文件中是没有任何问题的,相信每个初学者都会遇到这样的应用场景:使用一个 db.php 来定义数据库连接,在其他文件中直接引用这个文件,达到代码复用的效果。

再让我们考虑这样一个应用场景,如果我们使用一个框架,或者一个第三方包,里面一般有成百上千个类文件,而我们通常是不用自己去加载这些文件的,此时用的便是php的自动加载机制

定义一种自动加载模式

在传统的应用中,通常自定义 __autoload()。如下

define("dir", "/var/www/myweb/myclass/");
function __autoload($classname) {
  require dir.$classname.'.class.php';
}
$book = new book();

上述代码运作过程如下:

1. 自定义 __autoload 函数,它定义了类文件的加载方式
2. 当我们 new 一个 book 实例时,它首先看当前是否包含了这个类,如果不存在则自动调用 __autoload 函数并将类名 book 作为参数传递给这个函数。这实际上就是一种动态加载的方式,只有我们需要的类文件才会被加载。
3. 找到 __autoload 函数后,发现定义好的加载动作 require dir.$classname.'.class.php'; 这时候它就会去 dir 目录下查找 book.class.php 文件,如果存在这个文件则加载。
4. 关于类 book.class.php 的定义必须满足如下条件:类名和文件名一致;一个文件只定义一个类。

book.class.php 文件如下

class book {
  public function __construct() {
    echo "this is book's construct\n";
  }
}

注册多种加载模式

对于我们自己的简单应用,一种加载模式可能够用了,但是对于较大型的应用,上面的方式存在明显的缺陷:__autoload函数不能重复定义,也就是说我们只能定义一种加载文件的模式,最终的结果就是我们的类只能放在一个地方,这显然是不符合实际要求的。因此php使用了函数 spl_autoload_register 来代替 __autolaod

代码如下:

define("model_dir", "/var/www/myweb/mymodel/");
define("controller_dir", "/var/www/myweb/mycontroller/");
// 定义model类加载方式
function loadmodel($classname) {
  $filename = model_dir.$classname.'.php';
  if (file_exists($filename))
    require $filename;
}
// 定义controller加载方式
function loadcontroller($classname) {
  $filename = controller_dir.$classname.'.php';
  if (file_exists($filename))
    require $filename;
}
// 注册两个加载函数
spl_autoload_register("loadmodel");
spl_autoload_register("loadcontroller");
// 自动加载类文件
$bookmode = new bookmode();
$bookcontroller = new bookcontroller();

在上面的代码中,我们可以看到:

1. 可以使用任意函数名定义多个加载函数
2. 在 spl_autoload_register 对加载函数进行注册,实际上应该是添加到一个类似双向队列的数据结构中。
3. 当我们 new 的对象不存在于当前文件时,它会自动从我们的加载函数中查找,并且是按照我们使用 spl_autoload_register 注册的顺序进行的。
4. 需要注意的是,此时如果我们定义了 __autoload 方法,也必须进行注册,否则会被忽略。

spl_autoload_register三种注册函数的方式:

spl_autoload_register(funname); // 直接注册一个普通加载函数
spl_autoload_register(obj::method); // 注册一个静态加载方法
spl_autoload_regitser(array(obj, method)); // 当obj为类字符串时,只能加载静态方法。否则都可以。

实例

在各种php框架中,也大量用到了自动加载机制,我们通过laravel的一个小例子来看下。

laravel通过 ioc 容器帮我们管理依赖,让我们可以通过函数参数的方式愉快地获得了类实例,但我们也发现,我们并没有require文件,那容器又是如何找到我们的文件地址的?下面我们就来解决这个问题。

通过入口文件 index.php 我们一步步搜索,可以找到 /vendor/composer/classloader.php 文件。

部分代码如下

public static function loadclassloader($class)
{
  if ('composer\autoload\classloader' === $class) {
    require __dir__ . '/classloader.php';
  }
}
public static function getloader()
{
  if (null !== self::$loader) {
    return self::$loader;
  }
  spl_autoload_register(array('obj', 'loadclassloader'), true, true);
  // 通过命名空间的方式使用注册的加载类
  self::$loader = $loader = new \composer\autoload\classloader();
  spl_autoload_unregister(array('obj', 'loadclassloader'));
  if (php_version_id >= 50600) {
    // 该文件定义了包类和用户类的命名空间和实体文件的映射
    // 以及其他一些东西
    require_once __dir__ . '/autoload_static.php';
    // 初始化$loader一些属性。
    // 我们关注autoload_static.php文件的类映射
    // 被赋值在了 $loader的$classmap属性
        call_user_func(\composer\autoload\composerstaticinit::getinitializer($loader));
    // ...
  }
  // ...
  $loader->register(true);
  // ...
  return $loader;
}

它调用了 getloader() 函数,并将 loadclassloader 函数注册到加载函数注册队列。然后就可以通过命名空间的方式 self::$loader = $loader = new \composer\autoload\classloader(); 实例化 classloader 类。

紧接着,他载入了 /autoload_static.php 文件,大致内容如下

// 里面还定义了包类和psr的一些标准
public static $classmap = array (
  'app\\common\\collection' => __dir__ . '/../..' . '/app/common/collection.php',
  'app\\common\\mgdb' => __dir__ . '/../..' . '/app/common/mgdb.php',
  'app\\common\\redis' => __dir__ . '/../..' . '/app/common/redis.php',
)

看到这里笔者兴奋了,因为上面的 collection, redis 正是笔者定义的类!
然后就是我们在laravel经常听到的一个名词 “register”。查看 classloader 类的 register 方法如下:

public function register($prepend = false)
{
  spl_autoload_register(array($this, 'loadclass'), true, $prepend);
}
public function loadclass($class)
{
  if ($file = $this->findfile($class)) {
    includefile($file);
    return true;
  }
}
public function findfile($class)
{
  // ...
  // class map lookup
  if (isset($this->classmap[$class])) {
    return $this->classmap[$class];
  }
  // ...
}

上面的register方法同样使用了自动加载机制。并将通过findfile函数和$classmap数组直接找到对应的类的具体位置。这也就是我们不用自己去加载类文件的原因 – 当我们实例化一个代码中找不到的类时,它便会在这里加载对应的类。

看到这里我们也发现了它的使用和我们之前讲的并不完全一致,我们是注册函数是为了通过文件夹来寻找类,而laravel注册函数是为了注册一个映射数组然后直接调用(整了个映射文件三千多行。。。)具体为什么要这么做得等下次通读加载源码部分后再写一篇博文(本来只想找一个框架的例子,蜜汁尴尬)

笔者实例

两年前负责学校某个协会线上部分时,主要是做微信开发,因为时不时就要加一个新功能,所以如果用一般的方式写起来是比较痛苦的,但是用框架又有点大材小用。因为就使用了下面这种简单的方式:

require "./basic/init.php";
define('web_path', '');
//声明自动加载函数并注册,指示加载路径与加载方法
  function wechatautoload($class_name)
  {
    $file_road = './function/'.$class_name.'.class.php';
    if(file_exists($file_road))
    {
      require_once($file_road);
    }
  }
  spl_autoload_register('wechatautoload');
//----------------------------------------------

初始化好配置之后,我们注册了一个加载函数,以后每一个新功能都只要在 function 文件夹下新增一个文件即可,其他部分的改动很少或者根本不用(根据业务场景)。

参考:php: spl_autoload_register - manual

更多关于php相关内容感兴趣的读者可查看本站专题:《php面向对象程序设计入门教程》、《php数组(array)操作技巧大全》、《php基本语法入门教程》、《php运算与运算符用法总结》、《php字符串(string)用法总结》、《php+mysql数据库操作入门教程》及《php常见数据库操作技巧汇总

希望本文所述对大家php程序设计有所帮助。