yii2采用的基于namespace的autoload机制,我们从初始化来参看yii2的autoload机制的整个过程,详解yii的初始化过程
1.在入口文件index.php我们可以看到代码:
- require(__DIR__ . '/../../vendor/autoload.php');
2.我们打开这个文件:
- // autoload.php @generated by Composer
- require_once __DIR__ . '/composer' . '/autoload_real.php';
- return ComposerAutoloaderInit90def245ed1c6f870abec3fefcc03f88::getLoader();
可以看到加载了文件/vendor/composer/autoload_real.php,打开这个文件,我们可以发现,里面定义了一个php的class类:
ComposerAutoloaderInit90def245ed1c6f870abec3fefcc03f88
也就是调用了这个类的getLoader()方法。
3.找到这个方法getLoader()方法:下面是所有的代码:
- // autoload_real.php @generated by Composer
- class ComposerAutoloaderInit90def245ed1c6f870abec3fefcc03f88
- {
- private static $loader;
- public static function loadClassLoader($class)
- {
- if ('Composer\Autoload\ClassLoader' === $class) {
- require __DIR__ . '/ClassLoader.php';
- }
- }
- public static function getLoader()
- {
- if (null !== self::$loader) {
- return self::$loader;
- }
- spl_autoload_register(array('ComposerAutoloaderInit90def245ed1c6f870abec3fefcc03f88', 'loadClassLoader'), true, true);
- self::$loader = $loader = new \Composer\Autoload\ClassLoader();
- spl_autoload_unregister(array('ComposerAutoloaderInit90def245ed1c6f870abec3fefcc03f88', 'loadClassLoader'));
- $map = require __DIR__ . '/autoload_namespaces.php';
- foreach ($map as $namespace => $path) {
- $loader->set($namespace, $path);
- }
- $map = require __DIR__ . '/autoload_psr4.php';
- foreach ($map as $namespace => $path) {
- $loader->setPsr4($namespace, $path);
- }
- $classMap = require __DIR__ . '/autoload_classmap.php';
- if ($classMap) {
- $loader->addClassMap($classMap);
- }
- $loader->register(true);
- $includeFiles = require __DIR__ . '/autoload_files.php';
- foreach ($includeFiles as $file) {
- composerRequire90def245ed1c6f870abec3fefcc03f88($file);
- }
- return $loader;
- }
- }
- function composerRequire90def245ed1c6f870abec3fefcc03f88($file)
- {
- require $file;
- }
首先我们看到的是spl_autoload_register这个方法,这个方法的作用是,在找不到类的情况下,通过这个函数定义的类方法去找,通过传递的参数,返回加载的类的路径。也就是说,当找不到类的时候,就通过这个方法找:
- public static function loadClassLoader($class)
- {
- if ('Composer\Autoload\ClassLoader' === $class) {
- require __DIR__ . '/ClassLoader.php';
- }
- }
因此这段代码 self::$loader = $loader = new \Composer\Autoload\ClassLoader();
加载的文件是vendor/composer/ClassLoader.php
4.然后通过这个类的方法,加载很多初始路径:
- $map = require __DIR__ . '/autoload_namespaces.php';
- foreach ($map as $namespace => $path) {
- $loader->set($namespace, $path);
- }
- $map = require __DIR__ . '/autoload_psr4.php';
- foreach ($map as $namespace => $path) {
- $loader->setPsr4($namespace, $path);
- }
- $classMap = require __DIR__ . '/autoload_classmap.php';
- if ($classMap) {
- $loader->addClassMap($classMap);
- }
- $loader->register(true);
- $includeFiles = require __DIR__ . '/autoload_files.php';
- foreach ($includeFiles as $file) {
- composerRequire90def245ed1c6f870abec3fefcc03f88($file);
- }
- return $loader;
4.1通过set setPsr4 addClassMap等方法进行namespace路径初始化。 这个对应的文件是/autoload_psr4.php , 这个文件里面是对yii2的插件的namespace的定义:
- // autoload_psr4.php @generated by Composer
- $vendorDir = dirname(dirname(__FILE__));
- $baseDir = dirname($vendorDir);
- return array(
- 'yii\\swiftmailer\\' => array($vendorDir . '/yiisoft/yii2-swiftmailer'),
- 'yii\\redis\\' => array($vendorDir . '/yiisoft/yii2-redis'),
- 'yii\\gii\\' => array($vendorDir . '/yiisoft/yii2-gii'),
- 'yii\\faker\\' => array($vendorDir . '/yiisoft/yii2-faker'),
- 'yii\\debug\\' => array($vendorDir . '/yiisoft/yii2-debug'),
- 'yii\\composer\\' => array($vendorDir . '/yiisoft/yii2-composer'),
- 'yii\\codeception\\' => array($vendorDir . '/yiisoft/yii2-codeception'),
- 'yii\\bootstrap\\' => array($vendorDir . '/yiisoft/yii2-bootstrap'),
- 'yii\\' => array($vendorDir . '/yiisoft/yii2'),
- 'fecadmin\\' => array($vendorDir . '/fancyecommerce/fec_admin'),
- 'fec\\' => array($vendorDir . '/fancyecommerce/fec'),
- 'cebe\\markdown\\' => array($vendorDir . '/cebe/markdown'),
- 'Faker\\' => array($vendorDir . '/fzaninotto/faker/src/Faker'),
- );
定义各个插件的根路径。
4.2autoload_classmap.php 这个目前为空,没有细致研究具体内部的存放
4.3/autoload_files.php
- // autoload_files.php @generated by Composer
- $vendorDir = dirname(dirname(__FILE__));
- $baseDir = dirname($vendorDir);
- return array(
- '2cffec82183ee1cea088009cef9a6fc3' => $vendorDir . '/ezyang/htmlpurifier/library/HTMLPurifier.composer.php',
- '2c102faa651ef8ea5874edb585946bce' => $vendorDir . '/swiftmailer/swiftmailer/lib/swift_required.php',
- );
4.4 这是非yii2插件的包库 autoload_namespaces.php
- // autoload_namespaces.php @generated by Composer
- $vendorDir = dirname(dirname(__FILE__));
- $baseDir = dirname($vendorDir);
- return array(
- 'PHPExcel' => array($vendorDir . '/phpoffice/phpexcel/Classes'),
- 'Imagine' => array($vendorDir . '/imagine/imagine/lib'),
- 'HTMLPurifier' => array($vendorDir . '/ezyang/htmlpurifier/library'),
- 'Diff' => array($vendorDir . '/phpspec/php-diff/lib'),
- );
譬如我的yii2 – fec 插件中加入的PHPExcel 和Imagine 库包,就会在这里被标注namespace的对应关系。
对于composer安装的库包,有的是include的方式加入的,譬如Excel,安装库包后,不需要再程序中require,包管理器在autoload_namespaces.php 会加入路径,自动加载进来。
还有的是基于namespaces的,譬如Imagine。
5. 通过上面的配置,去找到对应文件路径,加载文件。
也就是说,对于 https://packagist.org/ 这里的php的库包,我们都可以通过composer加载到我们的系统中,在线安装。
譬如:我的fec插件的 composer.json的配置。
- "require": {
- "php": ">=5.4.0",
- "yiisoft/yii2": ">=2.0.6",
- "imagine/imagine": "0.5.*",
- "phpoffice/phpexcel": "1.8.*"
- },
- "autoload": {
- "psr-4": {
- "fec\\": ""
- }
- },
其中require代表的需要下载的包
autoload psr-4 里面添加了信息后,会在 vendor/yiisoft/extensions.php 文件中加入别名。
- 'fancyecommerce/fec' =>
- array (
- 'name' => 'fancyecommerce/fec',
- 'version' => '1.1.2.4',
- 'alias' =>
- array (
- '@fec' => $vendorDir . '/fancyecommerce/fec',
- ),
- ),
vendor/composer/autoload_psr4.php 中加入namespace信息:
- 'fecadmin\\' => array($vendorDir . '/fancyecommerce/fec_admin'),
- 'fec\\' => array($vendorDir . '/fancyecommerce/fec'),