自动加载
?php
或许你已经对自动加载有所了解。简单描述一下:自动加载
就是咱们在new一个class的时候,不须要手动去写require来导入这个class.php文件,程序自动帮咱们加载导入进来。这是php5.1.2(好像是)版本新加入一个功能,他解放了程序员的双手,不须要手动写那么多的require
,变得有那么点智能的感受。程序员
自动加载能够说是现代PHP框架的根基,任何牛逼的框架或者架构都会用到它,它发明出来的理由是啥呢?一个字:懒。由于项目越来愈大,相关联的类库文件愈来愈多,咱们不可能再像小项目那样在一个文件中所有手动一个一个require
。sql
如何才能自动加载呢? PHP 5.2版本更新了自动加载
须要的一个魔术方法——__autoload($class_name)
数组
正是这个神奇的内置魔术函数,才能让咱们这些屌丝偷懒。咱们来看下这个如何使用它。架构
自动加载的原理,就是在咱们new一个class
的时候,PHP系统若是找不到你这个类,就会去自动调用本文件中的__autoload($class_name)
方法,咱们new的这个class_name 就成为这个方法的参数。因此咱们就能够在这个方法中根据咱们须要new class_name的各类判断和划分就去require
对应的路径类文件,从而实现自动加载。app
咱们先一步步来,看下__autoload()的自动调用,看个例子:
index.php框架
$db =new DB();
若是咱们不手动导入DB类,程序可能会报错,说找不到这个类:ide
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()方法里,根据须要去加载类库文件了。ui
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();
小的项目,用__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就行了。
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
咱们分析下:
注意,前面说过,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)
前面已经说过,自动加载如今是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
好了。基本自动加载的东西就讲完了。很实用的东西。
在平时咱们使用命令空间时,有时候多是在同一个命名空间下的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这个空间下进行相对路径的加载,是错误的。