转载自:https://blog.csdn.net/baidu_30000217/article/details/52743139

php实现类文件自动载入有两种办法:

  1. 魔术方法:__autoload();
  2. SPL标准库提供的spl_autoload_register();

一、__autoload()

__autoload()是php中的一个魔术方法,在代码中当调用不存在的类时会自动调用该方法。

假如现在有以下两个类文件:

//Test1.php文件
<?php
class Test1{
static function test(){
echo "test1";
}
}
//Test2.php文件
<?php
class Test2{
static function test(){
echo "test2";
}
}

现在Test.php文件中要用到 Test1.php 和 Test2.php 中的类:

//Test.php文件
<?php
include "Test1.php";
include "Test2.php"; Test1::test();
Test2::test();

用 include 或require 的问题是当我要调用的类很多的时候,include 或 require也会很多,造成代码的冗杂,而且每次执行到 Test.php 文件的时候都要加载这么多文件,有些文件还不一定用到,那就浪费了很多内存,降低效率。再者是当你某个类文件被删掉了,你还得去修改 Test.php 文件。

由于这些原因,我们用 __autoload() 去代替 include

//Test.php文件
<?php
function __autoload($class){
if(file_exists($class.".php")){
require_once($class.".php");
}else{
die("文件不存在!");
}
} Test1::test();
Test2::test();

__autoload() 魔术方法的作用是当你调用不存在的类时会被自动调用,在 Test.php文件中我们调用 类Test1 和 类Test2,由于我们没有显式的引用类文件,那么系统就会自动调用 __autoload() 方法。

但是,到现在为止 __autoload() 方法基本上被废弃了!为啥呢?因为:

1、最大的缺陷就是一个文件中不允许有多个 __autoload()方法,想象一下,你的项目引用了别人的一个项目,你的项目中有一个 __autoload,别人的项目也有一个__autoload,这样两个__autoload就冲突了。解决的办法就是修改__autoload成为一个,这无疑是非常繁琐的。
2、假如你的项目中的类根据不同的用处放在不同的文件夹中 classes 和 core,然后Test.php文件中要分别调用里面对应的类,怎么搞?这样?

function __autoload($class){
if(file_exists("classes/".$class.".php")){
require_once("classes/".$class.".php");
}else{
die("文件不存在!");
}
}
function __autoload($class){
if(file_exists("core/".$class.".php")){
require_once("core/".$class.".php");
}else{
die("文件不存在!");
}
} Test1::test();
Test2::test();

这样做的话会出现致命错误,因为 __autoload()重复定义!(其实第1点的原因也是一样)

为了解决这个问题,于是就有 spl_autoload_register()

二、spl_autoload_register()

好,到了这一步,我们先不说 spl_autoload_register(),既然不用__autoload(),那么我们就自己定义负责类加载的函数:

function my_autoload1($class){
if(file_exists("classes/".$class.".php")){
require_once("classes/".$class.".php");
}else{
die("文件不存在!");
}
}
function my_autoload2($class){
if(file_exists("core/".$class.".php")){
require_once("core/".$class.".php");
}else{
die("文件不存在!");
}
} //现在我们就可以加载类文件啦
my_autoload1("Test1");
my_autoload2("Test2"); Test1::test();
Test2::test();

但是如上代码所示,直接调用我们的自定义类文件加载函数跟 include 有啥区别吗?这时 spl_autoload_register 就派上用场了。

显然,创造一个负责类文件加载函数不是为了让我们直接调用它,而是让PHP在需要类定义的时候为我们调用它。我们称这种功能为“自动加载”。

要开启“自动加载”功能,需要将加载函数注册到PHP中:

spl_autoload_register("my_autoload1");

我们重新实现 Test.php:

//Test.php文件
function my_autoload1($class){
if(file_exists("classes/".$class.".php")){
require_once("classes/".$class.".php");
}else{
die("文件不存在!");
}
}
function my_autoload2($class){
if(file_exists("core/".$class.".php")){
require_once("core/".$class.".php");
}else{
die("文件不存在!");
}
} //将加载函数注册到PHP中
spl_autoload_register("my_autoload1");
spl_autoload_register("my_autoload2"); Test1::test();
Test2::test();

这样就是实现了PHP的类文件自动载入功能。

__autolaod的更多相关文章

  1. 现代php编程

    自动加载__autolaod和spl_autoload_register() 自动加载就是指如果找不到某个类如何处理的方式,具体可参见此文,可以说spl_autoload_register是更加高级, ...

随机推荐

  1. Navicat 的安装及破解

    本地环境: ubutun 14 1.安装. ①先老实选择官方试用版安装(不推荐在各个下载平台去下载) 中文版:http://www.navicat.com.cn/download/navicat-fo ...

  2. NPM 使用及npm升级中问题解决

    NPM是随同NodeJS一起安装的包管理工具,能解决NodeJS代码部署上的很多问题,常见的使用场景有以下几种: 允许用户从NPM服务器下载别人编写的第三方包到本地使用. 允许用户从NPM服务器下载并 ...

  3. Oracle 拼接字符串

    Tom大神写的 CREATE OR REPLACE FUNCTION stragg(input varchar2) RETURN varchar2 PARALLEL_ENABLE AGGREGATE ...

  4. Windows与Linux的回车换行转换

    最初"\r"(return)表示“回车”即回到行首,“\n”(next)表示“换行”即定位到下一行:UNIX和Linux使用“\n”换行,而Windows用“\r\n”(不是\n\ ...

  5. error: http://ppa.launchpad.net lucid Release: The following signatures couldn't be verified because

    ubuntu 命令行sudo apt-get update W: GPG error: http://ppa.launchpad.net lucid Release: The following si ...

  6. POJ 2373 Dividing the Path(DP + 单调队列)

    POJ 2373 Dividing the Path 描述 农夫约翰的牛发现,在他的田里沿着山脊生长的三叶草是特别好的.为了给三叶草浇水,农夫约翰在山脊上安装了喷水器. 为了使安装更容易,每个喷头必须 ...

  7. R语言中的采样与生成组合

    不放回采样:sample(1:10, 5, replace = FALSE) 生成组合:

  8. java string字符拼接符"+"的研究

    程序: public class Test { public static void main(String args[]) { String s1 = "abc"; String ...

  9. netcore 2.0 部署 到iis

    .net Core2.0应用程序发布window服务器报错容易错过的配置. 1.应用程序发布. 2.IIS上新建网站. 3.应用程序池选择无托管代码. 4.服务器上安装DotNetCore.1.0.2 ...

  10. core1.1 升级到 2.0

    1.直接修改项目 1.1 改成 2.0 Startup 的修改 去除构造函数中下面的代码 var builder = new ConfigurationBuilder() .SetBasePath(e ...