今天就跟大家聊聊有关如何在PHP中实现自动加载机制,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。
创新互联拥有网站维护技术和项目管理团队,建立的售前、实施和售后服务体系,为客户提供定制化的成都做网站、网站建设、网站维护、服务器托管解决方案。为客户网站安全和日常运维提供整体管家式外包优质服务。我们的网站维护服务覆盖集团企业、上市公司、外企网站、商城网站定制开发、政府网站等各类型客户群体,为全球近1000家企业提供全方位网站维护、服务器维护解决方案。在传统的应用中,通常自定义__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'); //----------------------------------------------
看完上述内容,你们对如何在PHP中实现自动加载机制有进一步的了解吗?如果还想了解更多知识或者相关内容,请关注创新互联行业资讯频道,感谢大家的支持。