PHP中的自动加载操作实现方法详解


Posted in PHP onAugust 06, 2019

本文实例讲述了PHP中的自动加载操作实现方法。分享给大家供大家参考,具体如下:

what is 自动加载?

或许你已经对自动加载有所了解。简单描述一下:自动加载就是我们在new一个class的时候,不需要手动去写require来导入这个class.php文件,程序自动帮我们加载导入进来。这是php5.1.2(好像是)版本新加入一个功能,他解放了程序员的双手,不需要手动写那么多的require,变得有那么点智能的感觉。

自动加载可以说是现代PHP框架的根基,任何牛逼的框架或者架构都会用到它,它发明出来的理由是啥呢?一个字:懒。因为项目越来愈大,相关联的类库文件越来越多,我们不可能再像小项目那样在一个文件中全部手动一个一个require

如何才能自动加载呢? PHP 5.2版本更新了自动加载需要的一个魔术方法——__autoload($class_name)

正是这个神奇的内置魔术函数,才能让我们这些?潘客道痢N颐抢纯聪抡飧鋈绾问褂盟??/p>

1. 自动加载的原理以及__autoload的使用

自动加载的原理,就是在我们new一个class的时候,PHP系统如果找不到你这个类,就会去自动调用本文件中的__autoload($class_name)方法,我们new的这个class_name 就成为这个方法的参数。所以我们就可以在这个方法中根据我们需要new class_name的各种判断和划分就去require对应的路径类文件,从而实现自动加载。

我们先一步步来,看下__autoload()的自动调用,看个例子:

index.php

$db =new DB();

如果我们不手动导入DB类,程序可能会报错,说找不到这个类:

Fatal error: Class 'DB' not found in D:\wamp\www\testphp\autoload\index.php on line 3

那么,我们现在加入__autoload()这个方法再看看:

$db =new DB();
function __autoload($className)
{
  echo $className;
  exit();
}

根据上面自动加载机制的描述,你分析下会输出什么? 没错:肯定是输出:DB, 也就是我们需要new 的类的类名。所以,这个时候我们就可以在__autoload()方法里,根据需要去加载类库文件了。

index.php

$db =new DB();
function __autoload($className)
{
  require $className .'.php';
}

DB.php

class DB
{
  publicfunction __construct()
  {
      echo 'Hello DB';
  }
}

这样子我们就很轻松的将我们需要new 的class 全部导入了进来,这样子,我们就可以轻松的new N个class,比如:

<?php
function __autoload($className)
{
  require $className .'.php';
}
$db =new DB();
$info =newInfo();
$gender =newGender();
$name =newName();
//也是支持静态方法直接调用的
Height::test();

2. spl_autoload_register的使用

小的项目,用__autoload()就能实现基本的自动加载了。但是如果一个项目过大,或者需要不同的自动加载来加载不同路径的文件,这个时候__autoload就悲剧了,原因是一个项目中仅能有一个这样的 __autoload() 函数,因为 PHP 不允许函数重名,也就是说你不能声明2个__autoload()函数文件,否则会报致命错误,我了个大擦,那怎么办呢?放心,你想到的,PHP开发大神早已经想到。

所以spl_autoload_register()这样又一个牛逼函数诞生了,并且取而代之它。它执行效率更高,更灵活

先看下它如何使用吧:

当我们去new一个找不到的class时,PHP就会去自动调用sql_autoload_resister注册的函数,这个函数通过它的参数传进去:

sql_autoload_resister($param) 这个参数可以有多种形式:
sql_autoload_resister('load_function'); //函数名
sql_autoload_resister(array('load_object', 'load_function')); //类和静态方法
sql_autoload_resister('load_object::load_function'); //类和方法的静态调用
//php 5.3之后,也可以像这样支持匿名函数了。
spl_autoload_register(function($className){
  if (is_file('./lib/' . $className . '.php')) {
    require './lib/' . $className . '.php';
  }
});

index.php

function load1($className)
{
  echo 1;
  require $className .'.php';
}
spl_autoload_register('load1');//将load1函数注册到自动加载队列中。
$db =new DB();//找不到DB类,就会自动去调用刚注册的load1函数了

上面就是实现了自动加载的方式,我们同样也可以用类加载的方式调用,但是必须是static方法:

class autoloading {
//必须是静态方法,不然报错
  public static function load($className)
  {
    require $className .'.php';
  }
}
//2种方法都可以
spl_autoload_register(array('autoloading','load'));
spl_autoload_register('autoloading::load');
$db =new DB();//会自动找到

需要注意的是,如果你同时使用spl_autoload_register__autoload__autoload会失效!!! 再说了,本来就是替换它的,就一心使用spl_autoload_register就好了。

3. 多个spl_autoload_register的使用

spl_autoload_register是可以多次重复使用的,这一点正是解决了__autoload的短板,那么如果一个页面有多个,执行顺序是按照注册的顺序,一个一个往下找,如果找到了就停止。

我们来看下这个例子,DB.php就在本目录下,Info.php在/lib/目录下。

function load1($className)
{
  echo 1;
  if(is_file($className .'.php')){
    require $className .'.php';
  }
}
function load2($className)
{
  echo 2;
  if(is_file('./app/'. $className .'.php')){
    require'./app/'. $className .'.php';
  }
}
function __autoload($className)
{
  echo 3;
  if(is_file('./lib/'. $className .'.php')){
    require'./lib/'. $className .'.php';
  }
}
//注册了3个
spl_autoload_register('load1');
spl_autoload_register('load2');
spl_autoload_register('__autoload');
$db =new DB();//DB就在本目录下
$info =newInfo();//Info 在/lib/Info.php

我们注册了3个自动加载函数。执行结果是啥呢?

1Hello DB
123Hello Info

我们分析下:

new DB的时候,就按照注册顺序,先去找load1()函数了,发现找到了,就停止了,所以输出1 Hello Word

new Info的时候,先是安装注册顺序,先找load1(), 所以输出了1,发现没找到,就去load2()里面去找,所以输出了2,还是没这个文件,就去__autoload()函数里找,所以,先输出了3,再输出Hello Info

注意,前面说过,spl_autoload_register使用时,__autoload会无效,有时候,我们希望它继续有效,就可以也将它注册进来,就可以继续使用。

我们可以打印spl_autoload_functions()函数,来显示一共注册了多少个自动加载:

var_dump(spl_autoload_functions());
//数组的形式输出
array (size=3)
 0 => string 'load1' (length=5)
 1 => string 'load2' (length=5)
 2 => string '__autoload' (length=10)

4. spl_autoload_register自动加载+namespace命名空间 的使用

前面已经说过,自动加载现在是PHP现代框架的基石,基本都是spl_autoload_register来实现自动加载。namespace也是使用比较多的。所以spl_autoload_register + namespace 就成为了一个主流。根据PSR-0的规范,namespace命名已经非常规范化,所以用namespace就能找到详细的路径,从而找到类文件。

我们举例子来看下:

AutoLoading\loading

<?php
namespaceAutoLoading;
class loading {
  public static function autoload($className)
  {
    //根据PSR-O的第4点 把 \ 转换层(目录风格符) DIRECTORY_SEPARATOR ,
    //便于兼容Linux文件找。Windows 下(/ 和 \)是通用的
    //由于namspace 很规格,所以直接很快就能找到
    $fileName = str_replace('\\', DIRECTORY_SEPARATOR, DIR .'\\'. $className).'.php';
    if(is_file($fileName)){
      require $fileName;
    }else{
      echo $fileName .' is not exist';die;
    }
  }
}

上面就是一个自动加载的核心思想方法。下面我们就来spl_autoload_register来注册这个函数:

index.php

<?php
//定义当前的目录绝对路径
define('DIR', dirname(__FILE__));
//加载这个文件
require DIR .'/loading.php';
//采用`命名空间`的方式注册。php 5.3 加入的
//也必须是得是static静态方法调用,然后就像加载namespace的方式调用,注意:不能使用use
spl_autoload_register("\\AutoLoading\\loading::autoload");
// 调用三个namespace类
//定位到Lib目录下的Name.php
Lib\Name::test();
//定位到App目录下Android目录下的Name.php
App\Android\Name::test();
//定位到App目录下Ios目录下的Name.php
App\Ios\Name::test();

由于我们是采用PSR-O方式来定义namespace的命名的,所以很好的定位到这个文件的在哪个目录下了。很爽。对不对。

APP\Android\Name

<?php
namespaceApp\Android;
className
{
  public function __construct()
  {
    echo __NAMESPACE__ ."<br>";
  }
  public static function test()
  {
    echo __NAMESPACE__ .' static function test <br>';
  }
}

所以就会很容易找到文件,并输出:

Lib static function test
App\Android static function test
App\Ios static function test

好了。基本自动加载的东西就讲完了。很实用的东西。

4. 同命名空间下的相互调用

在平时我们使用命令空间时,有时候可能是在同一个命名空间下的2个类文件在相互调用。这个时候就要注意,在自动调用的问题了。

比如Lib\Factory.php 和 Lib\Db\MySQL.php

我想在 Lib\Factory.php 中调用 Lib\Db\MySQL.php。怎么调用呢?以下是错误的示范:

newLib\Db\MySQL();
//报错,提示说 D:\wamp\www\testphp\module\Lib\Lib\Db\MySQL.php is not exist

看到没?这种方式是在Lib\命名空间的基础上来加载的。所以会加载2个Lib。这种方式相当于相对路径在加载。

正确的做法是,如果是在同一个命名空间下平级的2个文件。可以直接调用,不用命名空间。

newMySQL();//直接这样就可以了。
newDb\MySQL();//如果有个Db文件夹,就这样。

还有一种方法就是使用 use 。使用user就可以带上Lib了。use使用的是绝对路径。

useLib\Db\MySQL;
newMySQL();

我想在 Lib\Db\MySQL.php 中调用 Lib\Register.php。怎么调用呢?

应该这样

useLib\Register;
Register::getInstance();

因为现在已经在Lib\Db这样一个命名空间了,如果你不用use,而是使用Lib\Register::getInstance()或者使用Register::getInstance()的话。将是在Lib\Db这个空间下进行相对路径的加载,是错误的。

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

PHP 相关文章推荐
PHP HTML代码串 截取实现代码
Jun 29 PHP
ajax php 实现写入数据库
Sep 02 PHP
理解php原理的opcodes(操作码)
Oct 26 PHP
PHP学习之输出字符串(echo,print,printf,print_r和var_dump)
Apr 17 PHP
PHP-redis中文文档介绍
Feb 07 PHP
深入理解PHP几个算法:PHP冒泡、PHP二分法、PHP求素数、PHP乘法表
Jun 06 PHP
PHP使用递归方式列出当前目录下所有文件的方法
Jun 02 PHP
PHP封装的Twitter访问类实例
Jul 18 PHP
详解YII关联查询
Jan 10 PHP
浅谈PHP中静态方法和非静态方法的相互调用
Oct 04 PHP
Laravel5.1 框架表单验证操作实例详解
Jan 07 PHP
php设计模式之观察者模式实例详解【星际争霸游戏案例】
Mar 30 PHP
Thinkphp自定义生成缩略图尺寸的方法
Aug 05 #PHP
thinkphp5.1框架中容器(Container)和门面(Facade)的实现方法分析
Aug 05 #PHP
RSA实现JS前端加密与PHP后端解密功能示例
Aug 05 #PHP
thinkPHP5框架接口写法简单示例
Aug 05 #PHP
ThinkPHP5+UEditor图片上传到阿里云对象存储OSS功能示例
Aug 05 #PHP
PHP各种常见经典算法总结【排序、查找、翻转等】
Aug 05 #PHP
php时间戳转换代码详解
Aug 04 #PHP
You might like
PHPMailer邮件发送的实现代码
2013/05/04 PHP
解放web程序员的输入验证
2006/10/06 Javascript
js getBoundingClientRect() 来获取页面元素的位置
2010/11/25 Javascript
jquery中邮箱地址 URL网站地址正则验证实例代码
2013/09/15 Javascript
JS函数的几种定义方式分析
2015/12/17 Javascript
详解前端自动化工具gulp自动添加版本号
2016/12/20 Javascript
Angular 5.x 学习笔记之Router(路由)应用
2018/04/08 Javascript
当vue路由变化时,改变导航栏的样式方法
2018/08/22 Javascript
基于Vue-cli快速搭建项目的完整步骤
2018/11/03 Javascript
vue.js的vue-cli脚手架中使用百度地图API的实例
2019/01/21 Javascript
vue-cli3+typescript新建一个项目的思路分析
2019/08/06 Javascript
[01:52]2020年DOTA2 TI10夏季活动预告片
2020/07/15 DOTA
Python格式化压缩后的JS文件的方法
2015/03/05 Python
python按照多个字符对字符串进行分割的方法
2015/03/17 Python
Python编写一个闹钟功能
2017/07/11 Python
基于python内置函数与匿名函数详解
2018/01/09 Python
Python设计模式之命令模式简单示例
2018/01/10 Python
人脸识别经典算法一 特征脸方法(Eigenface)
2018/03/13 Python
Python字符串、整数、和浮点型数相互转换实例
2018/08/04 Python
python3+PyQt5 数据库编程--增删改实例
2019/06/17 Python
py-charm延长试用期限实例
2019/12/22 Python
Python任务自动化工具tox使用教程
2020/03/17 Python
Django静态资源部署404问题解决方案
2020/05/11 Python
一款纯css3实现的动画加载导航
2014/10/08 HTML / CSS
英国领先的汽车轮胎和快速健康中心:Kwik Fit
2017/10/29 全球购物
澳大利亚汽车零部件、音响及配件超市:Automotive Superstore
2018/06/19 全球购物
艺术设计专业个人求职信
2013/09/21 职场文书
应用数学专业求职信
2014/03/14 职场文书
最常使用的求职信
2014/05/25 职场文书
宣传普通话标语
2014/06/27 职场文书
党员作风建设整改方案
2014/10/27 职场文书
开会迟到检讨书范文
2015/05/06 职场文书
务工证明怎么写
2015/06/18 职场文书
仓库管理制度范本
2015/08/04 职场文书
2016年清明节网上祭英烈活动总结
2016/04/01 职场文书
排查并解决Oracle sysaux表空间异常增长
2022/04/20 Oracle