下面由composer教程栏目给大家深入 Composer autoload,希望对需要的朋友有所帮助!众所周知 composer 是现代 PHP 项目的基石, 与古老的 pear 不同, composer 并不是一款专注于系统级别...
|
下面由composer教程栏目给大家深入 Composer autoload,希望对需要的朋友有所帮助!
这几天看到 phphub 上面有人开始进坑怒看 laravel 源代码,于是我也凑个热闹来看下这个故事。 众所周知 好吧,可能这是好事,但是也是坏事。好事是很多优秀的包都从 扯远了,扯回 自动加载的类型总体来说 composer 提供了几种自动加载类型
这几种自动加载都会用到,理论上来说,项目代码用 classmap这应该是最最简单的 autoload 模式了。大概的意思就是这样的: {
"classmap": ["src/"]
}然后 composer 在背后就会读取这个文件夹中所有的文件 然后再 <?php return [ 'App\\Console\\Kernel' => $baseDir . '/app/Console/Kernel.php' ]; ?> 然后就可以光明正大地用 好吧 上面的例子其实有点 tricky 就是上面这个 autoload 实际上是根据 prs-4 来生成出来的。不过这不重要,了解底层重要点,我们可以看到所有的所谓的 autoloading 其实可以理解为生成了这么一个 psr-0现在这个标准已经过时了。当初制定这个标准的时候主要是在 php 从 5.2 刚刚跃迁到 5.3+ 有了命名空间的概念。所以这个时候 {
"name": "acme/util",
"auto" : {
"psr-0": {
"Acme\\Util\\": "src/"
}
}
}文档结构是这样的 vendor/
acme/
util/
composer.json
src/
Acme/
Util/
ClassName.phpClassName.php 中是这样的 <?php
class Acme_Util_ClassName{}
?>我们可以看到由于旧版本的 php 没有 namespace 所以必须通过 这样稍微有点麻烦。指向一个文件夹之后 所以在 php5.2 版本已经彻底被抛弃的今天, psr-4这个标准出来的时候一片喷声,大概的意思就是 抛弃了 psr-0 的 composer 从此变得非常清爽。 最简单来讲就是可以把 prs-4 的 namespace 直接想想成 file structure {
"name": "acme/util",
"auto" : {
"psr-4": {
"Acme\\Util\\": "src/"
}
}
}vendor/
acme/
util/
composer.json
src/
ClassName.php可以看到将 <?php
namespace Acme\Util;
class ClassName {}
?>file然而这还是不够。因为可能会有一些全局的 helper function 的存在。 这个写法很简单就不多看了。 {
"files": [
"path/to/file.php"
]
}autoload_real.php好了看了所有的 autoload 类型那么直接怒看一发实现。 首先映入眼帘的就是一坨,我的是这样的
为啥? 因为这个类是全局的啊少年。 作为模块化大行其道的今天,全局的类总是有那么点奇怪。为了不让这个 autoload 的 helper 污染全局,composer 的仁兄们还是绞尽脑汁怒弄了这么一个奇怪的 hash。这直接就逼迫广大二笔程序员们不要跟这个撞衫。 好吧,接着往下看。 主要只有这么一个方法 <?php
// autoload_real.php @generated by Composer
class ComposerAutoloaderInit64c47026c93126586e44d036738c0862
{
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('ComposerAutoloaderInit64c47026c93126586e44d036738c0862', 'loadClassLoader'), true, true);
self::$loader = $loader = new \Composer\Autoload\ClassLoader();
spl_autoload_unregister(array('ComposerAutoloaderInit64c47026c93126586e44d036738c0862', '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) {
composerRequire64c47026c93126586e44d036738c0862($file);
}
return $loader;
}
}
function composerRequire64c47026c93126586e44d036738c0862($file)
{
require $file;
}
?>在讲什么?其实很简单。
其中 调查 Composer\ClassLoader 发现了这么一个注释
好吧还是怕二笔程序员犯浑。想想一下,如果有人在 autoload_files 中的文件中写了 不容易,终于快要胜利了。 为什么总是要 |
