今天写一个demo,遇到一个问题,平时框架用多了,下意识的就以为use就能引入类,导致一直跑不成功,后来才意识到use与引用类并不是一回事。use只是指定了要使用哪个命名空间下的类,但是并不会引入类,类的引用还是需要使用include或require。这里记录一下这次事件的总结,加深一下印象。
类的引用
创建一个类文件ClassA.class.php
namespace AreaA;
class ClassA{}
在同级目录下的index.php中要使用哪个类,就必须包含这个类文件,否则会报错
require('./AreaA/ClassA.class.php'); //引入AreaA下的ClassA类
$objA = new ClassA(); //实例化成功,实例化出来的时AreaA下的classA
use与类的引用
再创建另一个类文件ClassA.class.php,声明不同的命名空间
namespace AreaB;
class ClassA{}
在index.php中包含两个类文件
require('./AreaA/ClassA.class.php');
require('./AreaB/ClassA.class.php');
//此时单纯实例化ClassA时PHP不知道你用的是哪个ClassA,所以需要使用use来指定要使用的ClassA
use AreaA\ClassA;
$obj = new ClassA(); //此时实例化的是AreaA下的ClassA类
当然,不使用use也是可以的,但是在实例化的时候需要指定命名空间,如下:
require('./AreaA/ClassA.class.php');
require('./AreaB/ClassA.class.php');
$objA = new \AreaA\ClassA();
$objB = new \AreaB\ClassB();
类的自动加载
那这样每次使用的类的时候都要require一下类文件,岂不是很麻烦?框架中使用类的时候可没有每次都要手动包含一下类文件啊。
这就要说说类的自动加载了,框架不用每次都手动包含类文件就是因为php的类的自动加载功能,能够根据你使用的类名自动加载类文件。
php的自动加载有两种方式,__autoload()
函数与spl_autoload_register()
函数。
__autoload()
还是在index.php文件中
function __autoload($class_name){
$file = $class_name . '.class.php';
if (file_exists($file)) {
require_once($file);
}
}
use AreaA\ClassA;
$obj = new ClassA();
ps:在声明了命名空间后,自动加载函数获取的就是包含命名空间的类名,因此,将命名空间与文件名对应,对于自动加载来说会方便很多,可以自行在__autoload()
方法中将$class_name
打印出来看一看。
spl_autoload_register()
还是在index.php中
//声明一个自定义的加载类文件的函数
function _autoload($class_name){
$file = './' . $class_name . '.class.php';
$file = str_replace('\\','/',$file);
if (file_exists($file)) {
require_once($file);
}
}
//注册自定义的加载函数,注册之后,自定义的加载函数就变成了自动加载的函数
spl_autoload_register('_autoload');
use AreaA\ClassA;
$obj = new ClassA();
以上。