1、手动加载方式
像C和C++等语言,在PHP中需要使用另一个文件中的相关的类、方法时,可以使用include、include_once、require或者require_once将所用的文件包含进工程里面。
其中,四者的区别如下。
include将引用一个文件,如果文件不存在,则给出一个提示,跳过继续执行;
include_once也是套用一个文件,但是只会套用一次,如果文件不存在,则继续执行;
require表示引用一个文件,如果文件不存在,则中断程序的执行;
require_once也是套用一个文件,且只会套用一次,如果文件不存在,则中断程序的执行;
以上四种方式是需要什么文件的时候,手动在程序当中包含进文件。这在项目的规模比较小的时候,是可以的;但是随着项目规模的扩大,要通过手动的方式加载每个文件所需要的类简直是一场噩梦。
为了省事,在加载的时候可以通过set_include_path()设置加载的路径,同样也可以通过get_include_path()获取加载的路径。关set_include_path()get_include_path(),这里只对set_include_path()作简要的介绍。首先,set_include_path()是在脚本中动态的对php.ini中的include_path进行动态的修改,而这个include_path就是include和require的路径进行设置,或者说是预定义。
假如,我们在一个main.php文件中需要使用projname/lvfk/webapi/test文件夹下的a.php、b.php、c.php......,如果没有设置包含的路径的话,那么写成如下的形式:
A定义在A.php中
2.2、spl_autoload_register
SPL是Standard PHP Library(标准PHP库)的缩写,是PHP5引入的一个扩展库。SPL autoload是通过将函数指针autoload_func指向自动装载函数实现的。SPL具有两个不同的自动装载函数,分别是spl_autoload和spl_autoload_call,通过将autoload_fun指向这两个不同的加载函数地址可以实现不同的自动加载机制。
spl_autoload是SPL实现的默认的自动加载函数,是一个可以接受两个参数的函数。其中第一个函数为$class_name,表示要加载的类名;第二个参数是$file_extension为可选参数,表示类文件的扩展名。$file_extension中可以指定多个扩展名,扩展名之间用分号隔开即可,不指定扩展名,则使用默认的扩展名.inc或者.php。spl_autoload首先将$class_name变为小写,然后在所有的include_path中搜索$ class_name.inc或者$class_name.php文件。如果找到对应的文件,就加载对应的类。其实可以手动的使用spl_autoload("xxxx",".php")来实现xxxx类的加载。这其实和require/include差不多,但是,spl_autoload相对来说灵活一点,因为可以指定多个扩展名。
前面说到,spl_autoload_register中包含的函数指针autoload_func用于指定要使用的加载函数。那么,我们必须将对应的函数地址赋值给autoload_func,spl_autoload_register()正实现了给函数指针autoload_func赋值的功能。如果spl_autoload_register()函数中不含有任何的参数,则默认是将spl_autoload()赋值给autoload_func.
SPL模块的内部其实还存在着一个autoload_functions,其本质上是一个哈希表,或者为了直观的理解,我们将其想像成一个容器,里面的各个元素都是指向加载函数的指针。spl_autoload_call的实现机制其实也比较简单,按照一定的顺序遍历这个容器,执行里面的函数指针指向的加载函数,每执行一个指针之后都会检查所需要的类是否已经完成加载。如果完成了加载,则退出。否则继续接着向下执行。如果执行完所有的加载函数之后,所需要的类仍然没有完成加载,则spl_autoload_call()直接退出。这也就是说即使使用了autoload机制,也不一定能够完成类的加载,其关键在于看你如何创建你的自动加载函数。
既然,存在一个autoload_functions,那么如何将创建的自动加载函数添加到其中呢?spl_autoload一样,同样使用spl_autoload_register()将加载函数注册到autoload_functions中。当然可以通过spl_autoload_unregister()函数将已经注册的函数从autoload_functions从哈希表中删除。这和前面所写过的工厂设计模式是一致的,详见:http://www.cnblogs.com/yue-blog/p/5771352.html。
这里需要说明的一点是spl_autoload_register实现自动加载的顺序。spl_autoload的自动加载顺序为:首先判断autoload_func是否为空,如果autoload_func为空,则查看是否定义了__autoload函数,如果没有定义,则返回,并报错;如果定义了__autoload()函数,则返回加载的结果。如果autoload_func不为空,直接执行autoload_func指针指向的函数,不会检查__autoload是否定义。也就是说优先使用spl_autoload_register()注册过的函数。
根据以上介绍,如果autoload_func为非空是就不能自动执行__autoload()函数了。如果想在使用spl_autoload_register()函数的情况下,依然可以使用__autoload()函数,则可以将__autoload函数通过spl_autoload_register()添加到哈希表中,即,spl_autoload_register(__autoload())
以上摘取网友博客内容,介绍spl_autoload_register的工作流程回归主题,我们在小项目,用__autoload()就能实现基本的自动加载了,
但是如果一个项目过大,或者需要不同的自动加载来加载不同路径的文件,这个时候__autoload就悲剧了,
原因是一个项目中仅能有一个这样的 __autoload() 函数,因为 PHP 不允许函数重名,
也就是说你不能声明2个__autoload()函数文件,否则会报致命错误
所以spl_autoload_register()这样又一个牛逼函数诞生了,并且取而代之它。它执行效率更高,更灵活sql_autoload_resister($param) 这个参数可以有多种形式:
sql_autoload_resister('load_function'); //函数名
sql_autoload_resister(array('load_object', 'load_function')); //类和静态方法
sql_autoload_resister('load_object::load_function'); //类和方法的静态调用
spl_autoload_register是可以多次重复使用的,这一点正是解决了__autoload的短板,
那么如果一个页面有多个,执行顺序是按照注册的顺序,一个一个往下找,如果找到了就停止
2.4、spl_autoload_functions
array (size=2)
0 => string 'autoloading' (length=11)
1 => string 'load' (length=4)
3、spl_autoload_register搭配命名空间使用
自动加载现在是PHP现代框架的基石,基本都是spl_autoload_register来实现自动加载。namespace也是使用比较多的。所以spl_autoload_register + namespace 就成为了一个主流
示例:
IAutoloader.php
4、同命名空间下的相互调用
在平时我们使用命令空间时,有时候可能是在同一个命名空间下的2个类文件在相互调用。这个时候就要注意,在自动调用的问题了。
比如Lib\Factory.php 和 Lib\Db\MySQL.php
我想在 Lib\Factory.php 中调用 Lib\Db\MySQL.php。怎么调用呢?以下是错误的示范:
new Lib\Db\MySQL();
//报错,提示说 D:\wamp\www\testphp\module\Lib\Lib\Db\MySQL.php is not exist
new MySQL(); //直接这样就可以了。
new Db\MySQL(); //如果有个Db文件夹,就这样。
use Lib\Db\MySQL;
new MySQL();
use Lib\Register;
Register::getInstance();