オートローダーが必要な小さなアプリケーションがあります。symfony2 クラスローダーを簡単に使用できましたが、やり過ぎのようです。
安定した超軽量の psr-0 オートローダーはありますか?
あなたは非常に軽量であると尋ねます。そうしましょう;)
Timothy Boronczyk が素晴らしい最小 SPL オートローダーを作成しました: http://zaemis.blogspot.fr/2012/05/writing-minimal-psr-0-autoloader.html
次のようにコードを要約しました。
function autoload1( $class ) {
preg_match('/^(.+)?([^\\\\]+)$/U', ltrim( $class, '\\' ), $match ) );
require str_replace( '\\', '/', $match[ 1 ] )
. str_replace( [ '\\', '_' ], '/', $match[ 2 ] )
. '.php';
}
次に、この [autoload3] と短い @Alix Axel コード[autoload4] (の縮小版) を比較します。
function autoload3($c){preg_match('/^(.+)?([^\\\\]+)$/U',ltrim($c,'\\'),$m);require str_replace('\\','/',$m[1]).str_replace(['\\','_'],'/',$m[2]).'.php';}
function autoload4($c){require (($n=strrpos($c=ltrim($c,'\\'),'\\'))!==false?str_replace('\\','/',substr($c,0,++$n)):null).str_replace('_','/',substr($c,$n)).'.php';}
autoload3 が最短です。
安定した非常に軽量な (175b !) オートローダ ファイルを使用しましょう:
<?php spl_autoload_register(function ($c){preg_match('/^(.+)?([^\\\\]+)$/U',ltrim($c,'\\'),$m);require str_replace('\\','/',$m[1]).str_replace(['\\','_'],'/',$m[2]).'.php';});
たぶん私はクレイジーだけど、あなたはエクストリームを求めたのね?
EDIT: Alix Axelのおかげで、コードを短縮し(100bのみ!)、古いライブラリのさまざまなオートロード戦略がある場合に備えて、requireの代わりにincludeを使用しました(そして、splオートロードスタックのさまざまなオートローダー...)。
<?php spl_autoload_register(function($c){@include preg_replace('#\\\|_(?!.+\\\)#','/',$c).'.php';});
より短く/より良くしたい場合は、この要点を使用してください。
PSR-0 仕様書には、互換性のあるオートローダ機能の例があり、これはすでにかなり短いものです。
function autoload($className)
{
$className = ltrim($className, '\\');
$fileName = '';
$namespace = '';
if ($lastNsPos = strripos($className, '\\')) {
$namespace = substr($className, 0, $lastNsPos);
$className = substr($className, $lastNsPos + 1);
$fileName = str_replace('\\', DIRECTORY_SEPARATOR, $namespace) . DIRECTORY_SEPARATOR;
}
$fileName .= str_replace('_', DIRECTORY_SEPARATOR, $className) . '.php';
require $fileName;
}
その使用法は非常に簡単です。
spl_autoload_register('autoload');
include_path
欠点は、ディレクティブで動作するベースディレクトリを構成する必要があることです。SPL セマンティクスに基づいたハイブリッド PSR-0 オートローダーをサポートするために、次のオートローダーはインクルード パスと spl オートロード拡張機能をサポートしています。
$spl_autoload_register_psr0 = function ($extensions = null)
{
$callback = function ($className, $extensions = null)
{
if (!preg_match('~^[a-z0-9\\_]{2,}$~i', $className)) {
return;
}
null !== $extensions || $extensions = spl_autoload_extensions();
$extensions = array_map('trim', explode(',', $extensions));
$dirs = array_map('realpath', explode(PATH_SEPARATOR, get_include_path()));
$classStub = strtr($className, array('_' => '/', '\\' => '/'));
foreach ($dirs as $dir) {
foreach ($extensions as $extension) {
$file = sprintf('%s/%s%s', $dir, $classStub, $extension);
if (!is_readable($file)) {
continue;
}
include $file;
return;
}
}
};
return spl_autoload_register($callback);
};
Symfony2 ClassLoader コンポーネントには、ここでより多くの構成を許可するという利点があります。Pear または Composer ( Packagist の symfony/class-loader )を介して簡単にインストールできます。これは、多くのユーザーに使用され、十分にテストおよびサポートされている独自のコンポーネントです。
SplClassLoaderは正しい選択のようです。PSR-0自身が提案した実装です。
@hakre が提供する回答とまったく同じです。
function autoload($class) {
$path = null;
if (($namespace = strrpos($class = ltrim($class, '\\'), '\\')) !== false) {
$path .= strtr(substr($class, 0, ++$namespace), '\\', '/');
}
require($path . strtr(substr($class, $namespace), '_', '/') . '.php');
}
$path = null;
別の値に変更するか、次のようにしてベースディレクトリを設定することもできます。
$paths = array
(
__DIR__ . '/vendor/',
__DIR__ . '/vendor/phunction/phunction.php',
);
foreach ($paths as $path)
{
if (is_dir($path) === true)
{
spl_autoload_register(function ($class) use ($path)
{
if (($namespace = strrpos($class = ltrim($class, '\\'), '\\')) !== false)
{
$path .= strtr(substr($class, 0, ++$namespace), '\\', '/');
}
require($path . strtr(substr($class, $namespace), '_', '/') . '.php');
});
}
else if (is_file($path) === true)
{
require($path);
}
}