首页 > 代码库 > thinkphp学习笔记9—自动加载

thinkphp学习笔记9—自动加载

1.命名空间自动加载

在3.2版本中不需要手动加载类库文件,可以很方便的完成自动加载。

系统可以根据类的命名空间自动定位到类库文件,例如定义了一个类Org\Util\Auth类:

namespace Org\Util;class Auth {}

保存到ThinkPHP/Library/Org/Util/Auth.class.php

这样我们就可以直接实例化了,

new \Org\Util\Auth();

实例化之后系统会自动加载

ThinkPHP/Library/Org/Util/Auth.class.php 

框架Liberary目录下的命名空间都可以自动识别和定位,如下

  1. Library 框架类库目录
  2. ├─Think 核心Think类库包目录
  3. ├─Org Org类库包目录
  4. ├─ ... 更多类库目录

Library目录下的子目录都是一个根命名空间,就是说Think,Org下的类都可以自动加载

new Think\Cache\Driver\File();new Org\Util\Auth();new Org\Io\File();

都可以自动加载对应的类库文件。

我们还可以在Liberary目录下面任意增加新的目录,就会自动注册成为一个新的根命名空间,如下:

‘AUTOLOAD_NAMESPACE‘ => array(    ‘My‘     => THINK_PATH.‘My‘,    ‘One‘    => THINK_PATH.‘One‘,)

配置了上面的AUTOLOAD_NAMESPACE之后可以实例化下面的类库

new My\Net\IpLocation();new One\Util\Log();

自动加载下面的类库文件

ThinkPHP/My/Net/IpLocation.class.php
ThinkPHP/One/Util/Log.class.php

如果命名空间不在Library目录下面,并且没有定义对应的AUTO_LOADNAMESPACE参数的话则会当做模块的命名空间自动加载,例如

new Home\Model\UserModel();new Home\Event\UserEvent();

这跳跃有点大,刚才讲的还是Liberary下的命名空间,现在又扯到Application下的。

由于ThinkPHP/Library目录下面不存在Home目录,也没在AUTOLOAD_NAMESPACE参数定义Home命名空间,所以就把Home当成模块命名空间来识别,所以会自动加载

Application/Home/Model/UserModel.class.phpApplication/Home/Event/UserEvent.class.php

这些命名空间貌似都是ThinkPHP\Liberary目录下的,这个目录下的东西一般不会修改的啊,不明白为什么没有说Application目录下的命名空间。

2.类库映射

定义了较多的命名空间效率会有所影响,可以给常用的类库定义类库映射,命名空间映射相当于给类文件定义了一个别名,例如:

Think\Think::addMap(‘Think\Log‘,THINK_PATH.‘Think\Log.php‘);Think\Think::addMap(‘Org\Util\Array‘,THINK_PATH.‘Org\Util\Array.php‘);

那这段应该写在哪里呢,文档没有说清楚。也可使用addMap方法批量导入类库映射定义,如下:

$map = array(‘Think\Log‘=>THINK_PATH.‘Think\Log.php‘,‘Org\Util\Array‘=>THINK_PATH.‘Org\Util\Array.php‘);Think\Think::addMap($map);

文档中依旧没有说明这段应该写在什么地方,是config.php还是入口文件里呢,捉急啊。

3.类库加载的优先级

在实际的类库加载过程中,往往会涉及到自动加载优先级的问题,以Test\MyClass为例,自动加载的优先级顺序如下:

1.判断是否注册了Test\MyClass类库映射,如果有则自动加载类库映射定义的文件,

2.判断是否存在Liberary\Test目录,有则以该目录为初始目录加载

3.判断是否注册了Test根命名空间,有则以注册的目录为初始目录加载

4.以上都不成立则以Test模块模块经行初始目录加载,

 

4.手动加载第三方类库

使用第三方类库的时候可能出现以下情况,不符合ThinkPHP命名空间和后缀,没有使用命名空间或者命名空间和路径不一致,我们可以使用手动导入的方式加载。

我们使用import方法导入类库,如下:

// 导入Org类库包Library/Org/Util/Date.class.php类库import("Org.Util.Date");// 导入Home模块下面的 Application/Home/Util/UserUtil.class.php类库import("Home.Util.UserUtil");// 导入当前模块下面的类库 import("@.Util.Array");// 导入Vendor类库包 Library/Vendor/Zend/Server.class.phpimport(‘Vendor.Zend.Server‘);

对于import方法,系统会自动识别导入类库文件的位置,ThinkPHP可以自动识别的类库包括Think,Org,Com,Behavior,Vendor,以及Liberary目录下的子目录,这不扯犊子啊,Think,Org,Com,Behavior,Vendor就在ThinkPHP\Liberary下,文档中所说的Liberary目录是指那个目录呢?哦,有可能文档中所说的是Liberary目录下自己新建的目录,有可能。

如果在Liberary目录下新建了一个Test目录,并创建了一个UserTest.class.php文件可以这样导入:

import(‘Test.UserTest‘);

注意如果没有使用namespace来定义命名空间的话,实例化的时候需要使用根命名空间,如下:

import(‘Test.UserTest‘);$test = new \UserTest();

按照系统规则,import方法是无法导入具有点号的类库文件,因为点号会直接转化为斜线,例如如果我们定义了一个User.Info.class.php文件的话采用import("Org.User.Info");方式加载就会出现错误,导致我们加载到的不是Org/User.Info.class.php文件,而是Org/User/Info.class.php文件,这种情况下我们使用import("Org.User#Info");方式导入,这个就是说文件名中的点号用#代替。

 

大多数情况下import可以自动识别导入类库文件的位置,特殊情况下需要第二个参数来指定起始导入的路径,例如:要导入当前文件所在目录下的RBAC/AccessDecisionManager.class.php 文件,可以使用import("RBAC.AccessDecisionManager",dirname(__FILE__)); 。 如果要导入的文件后缀不是class.php而是.php,可以使用第三个参数import("RBAC.AccessDecisionManager",dirname(__FILE__),".php");

 

如果第三方的类库放在Vendor目录下,以.php为文件后缀,也没有采用命名空间可以采用系统内部的Vendor函数简化导入,例如我们要把Zend的Filter\Dir.php放到Vendor目录下面,这个时候Dir文件的路径就是Vendor\Zend\Filter\Dir.php,使用Vendor方法导入如下:

Vendor(‘Zend.Filter.Dir‘);

Vendor方法也可以支持和import一样的基础路径和文件后缀参数,如下:

Vendor(‘Zend.Filter.Dir‘,dirname(__FILE__),‘.class.php‘);

 

感觉高上大的,这些貌似是对系统进行扩展的时候使用的比较多。