PHP自动加载机制实例详解

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

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

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

PHP自动加载机制

定义一种自动加载模式

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

rush:PHP;"> 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 目录下查找 PHP 文件,如果存在这文件则加载。 4. 关于类 PHP 的定义必须满足如下条件:类名和文件名一致;一个文件只定义一个类。

Book.class.PHP 文件如下

rush:PHP;"> class Book { public function __construct() { echo "this is Book's construct\n"; } }

注册多种加载模式

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

代码如下:

rush:PHP;"> 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

三种注册函数的方式:

rush:PHP;"> 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 文件

部分代码如下

= 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 类。

紧接着,他载入了 PHP 文件,大致内容如下

__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 方法如下:

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注册函数是为了注册一个映射数组然后直接调用(整了个映射文件三千多行。。。)具体为什么要这么做得等下次通读加载源码部分后再写一篇博文(本来只想找一个框架的例子,蜜汁尴尬)

笔者实例

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

rush:PHP;"> 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程序设计有所帮助。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


服务器优化必备:深入了解PHP8底层开发原理
Golang的网络编程:如何快速构建高性能的网络应用?
Golang和其他编程语言的对比:为什么它的开发效率更高?
PHP8底层开发原理揭秘:如何利用新特性创建出色的Web应用
将字符重新排列以形成回文(如果可能)在C++中
掌握PHP8底层开发原理和新特性:创建高效可扩展的应用程序
服务器性能优化必学:掌握PHP8底层开发原理
PHP8新特性和底层开发原理详解:优化应用性能的终极指南
将 C/C++ 代码转换为汇编语言
深入研究PHP8底层开发原理:创建高效可扩展的应用程序
C++程序查找法向量和迹
PHP8底层开发原理实战指南:提升服务器效能
重排数组,使得当 i 为偶数时,arr[i] >= arr[j],当 i 为奇数时,arr[i] <= arr[j],其中 j < i,使用 C++ 语言实现
Golang的垃圾回收:为什么它可以减少开发人员的负担?
C++程序:将一个数组的所有元素复制到另一个数组中
Golang:构建智能系统的基石
为什么AI开发者应该关注Golang?
在C和C++中,逗号(comma)的用法是用来分隔表达式或语句
PHP8底层开发原理解析及新特性应用实例
利用PHP8底层开发原理解析新特性:如何构建出色的Web应用