PHP 命名空间 namespace / 类别名 use / 框架自动载入 机理的

相比 PHP5.2 版本 PHP5.3 新增了三大主要新特性php

命名空间 linux

延迟静态绑定 web

lambda匿名函数thinkphp

命名空间的出现也使PHP能够更加合理的组织项目结构,同时经过命名空间和自动载入机制一大批 PHP 的 MVC 框架也随之出现,明了的项目结构的同时也按需载入,进一步减轻内存压力,加快执行效率。bootstrap

由于命名空间是对目录结构友好的yii2

namespace Home\Controller;

class IndexController
{

}

而 PHP5.2 以前是按造类的下划线去作相似 命名空间 的定义的app

class Home_Controller_IndexController
{

}

 

1、 命名空间 及 USE 的本质

php 的 use 关键字并非马上导入所use的类,它只是声明某类的完整类名(命名空间::类标示符),然后你在上下文中使用此类时系统才会根据 use 声明获取此类的完整类名 而后利用自动加载机制进行载入框架

 

namespace Home\Controller;

use Home\Model\User;
use Home\Model\Order as OrderList;

class IndexController
{

    public function index()
    {
        //只有当你调用此类时,系统才会根据 use 声明获取此类的完整类名 而后利用自动加载机制进行载入
        $user = new User();
        $order = new OrderList();
    }
}

就像以下的代码 自动载入函数是在 use 两个类以后方才实现的 由于 use 并不会当即使用此类 只有在你调用此类时系统才会在找不到此类的状况下经过 autoload 函数动态延迟加载,若仍加载不到,则报错yii

use \Psr\Http\Message\ServerRequestInterface as Request;
use \Psr\Http\Message\ResponseInterface as Response;

require 'vendor/autoload.php';

$app = new \Slim\App();
$app->get('/hello/{name}', function (Request $request, Response $response) {
    $name = $request->getAttribute('name');
    $response->getBody()->write("Hello, $name");
    return $response;
});

$app->run();

2、剖析 Yii2 自动载入

先看一下 Yii2的控制器书写规则函数

controllers\IndexController.php

<?php 
/**
 * 自动载入机制
 */
//==========================================================================================
//声明本身的命名空间 此命名空间下的类皆归属于此命名空间管理
//==========================================================================================
namespace app\controllers;

//==========================================================================================
//use 声明其实只是为 yii\web\Controller 定义一个别名:Controller 方便咱们再上下文中使用
//而并不是像 C/C++的include JAVA的import将导入的文件加载进来(固然php为解释语言 不存在预编译这一步)
//==========================================================================================
use yii\web\Controller;

//==========================================================================================
//IndexController的完整类名其实为 app\controller\IndexController
//Controller 为 yii\web\Controller php 解释到此点才会去加载此类
//==========================================================================================
class IndexController extends Controller
{

    public function actionIndex()
    {
        //不使用 use 提早声明 手写完整的类名
        $model = new \app\models\EntryForm();//注意当前上下文中存在命名空间 因此要从根命名空间 '\' 开始
    }
}

一、某命名空间下的类 的完整名称为 namespace\className,当在某命名空间上下文中访问其它命名空间下的类时,咱们可使用 use 作别名化,或者使用此类的完整名称,但要以 '\' 根命名空间开头,不然解释器会认为你是在当前命名空间上下文中调用,即 foo\bar 方式会以 currentNamespace\foo\bar的方式去加载

命名空间与linux文件系统很类似,'\' 表明根,不以根开始的皆认为以当前命名空间为基点

二、use 只是给你使用的类定义短别名,use foo\bar 后则new bar() 即new  \foo\bar(),还有个小技巧,当咱们同时引用不一样命名空间下的类名相同的类时可使用 as 为其定义一个新别名

use foo\bar\sameName as classA;
use bar\foo\sameName as classB;

new classA(); // new \foo\bar\sameName;
new classB(); // new \bar\foo\sameName;

三、当咱们经过 入口文件 加载参数配置 实例化一个应用主体 加载路由组件解析请求 分派控制器调用方法时,期间会调用其余的类,好比 

use yii\web\Controller;

系统便会去经过自动载入函数作最一次载入尝试,若仍加载不到此类则报错

下面咱们看下 Yii2 从入口文件开始一个应用实体后注册自动载入函数的流程

index.php

入口文件载入配置和系统框架时会使用require调用,由于如今尚未注册自动加载函数

载入 Yii bootstrap 文件时便经过 spl_autoload_register 注册了自动载入函数 

Yii.php

Yii2的自动载入函数

继承至 BaseYii 它要作的就是根据你命名空间类型的类名去映射为此类所在的文件路径

好比 yii\web\Controller类会根据 yii 而映射到  YII2_PATH . '/web/Controller.php' 文件中,而这个文件则是命名空间为 yii\web 的 Controller 类,将此文件载入便可访问 yii\web\Controller

而咱们本身编写的控制器或者模型则访问时为 'app\controllers\IndexController' 'app\models\EntryForm'

则 autoload 函数会根据 app 为 映射关键字将其定位到 controllersmodels 文件夹下从而读取对应的文件便可载入相应的类,这也是为何 类名 与 文件名 相互对应的缘由所在,若不存在对应,则你只能经过固定的 require 某个文件去加载你写在其中的类了 

扩展本身的类库

咱们能够经过Yii2的自动载入机制灵活的归类咱们本身写的工具类等,好比我想建立一个本身的组件库

你能够定义一个  yii\tools 命名空间的类文件 MyTools.php,好比

<?php

namespace yii\tools;

class MyTools {
//.........
}
?>

放入 vendor\yiisoft\yii2\tools 文件夹下,

经过

namespace app\controllers;

// yii一级命名空间 则 映射到 YII_PATH 下
// 根据 tools\MyTools 定位到 YII_PATH 下的 tools文件夹下的 MyTools.php
use yii\tools\MyTools;
use yii\web\Controller;

class MyController extends Controller
{

}

固然你也能够在你的项目目录下新建一个 tools 文件夹 把 MyTools.php 放进去,将里面的命名空间改成 app\tools 便可,系统会根据 app 映射到项目根目录 经过 tools\MyTools 把 tools文件夹下的 MyTools.php文件载入 即载入了 MyTools 类

3、剖析TP的自动载入

thinkphp的自动加载规则也同样,只不过 tp autoload函数并无像 Yii2 basic 版预先定义一个项目根目录的映射规则,  Yii2则是以 app 顶级命名空间为默认的应用命名空间,yii顶级命名空间做为框架命名空间,因此你只要把本身的类归属到项目根目录(app下)或 YII_PATH(框架路径) 下,而后放对文件路径便可,

tp的话有的你本身想tp能够在 APP_PATH 下放多个  module ,像其预先定义的 Home ,或者你能够 BIND_MODULE来帮定义一个本身的模块,这样在经过入口文件载入的应用实体作路由时便能判断你请求的是哪一个模块下的控制器和方法

tp有几个系统占用的顶级命名空间

Think Org Behavior Com Vendor

而你本身的则会以 APP_PATH 为根目录进行加载,好比 Home\Controller\IndexController.class.php,当你访问 Index 时路由解析出来的类为 Home\Controller\IndexController,自动载入函数则根据 Home 非系统命名空间而定位到你的APP_PATH下进行加载,因此TP也能够本身定义的  AUTOLOAD_NAMESAPCE作自定义扩展

'AUTOLOAD_NAMESAPCE' => [
    'Tools' => APP_PATH . 'Vendor\Tools'
]

这样便把 Tools 顶级命名空间注册到了自动载入函数中,当咱们

use Tools\Extension\MyTools 时

传入 autoload 的 $class 即为 Tools\Extension\MyTools,获得的 $name 其实为一级命名空间名 这里为 Tools,Tools 不符合第一条件,在 else 中读取自定义的  AUTOLOAD_NAMESAPCE,发现咱们有设置 键名为 Tools 的成员

便使用 dirname(键值)获得 APP_PATH . 'Vendor',我是以为这里 dirname 写的有些鸡肋....因此便成功的映射定位出 Tools一级命名空间所在的文件目录为 APP_PATH . 'Vendor' 下,在与完整的类名 Tools\Extension\MyTools 拼接上 EXT便可定位到类文件,加载便可。

相关文章
相关标签/搜索