PHPのオートローディングを利用しようとしています。さまざまなディレクトリにさまざまなクラスがあるため、次のようにオートローディングをブートストラップしました。
function autoload_services($class_name)
{
$file = 'services/' . $class_name. '.php';
if (file_exists($file))
{
require_once($file);
}
}
function autoload_vos($class_name)
{
$file = 'vos/' . $class_name. '.php';
if (file_exists($file))
{
require_once($file);
}
}
function autoload_printers($class_name)
{
$file = 'printers' . $class_name. '.php';
if (file_exists($file))
{
require_once($file);
}
}
spl_autoload_register('autoload_services');
spl_autoload_register('autoload_vos');
spl_autoload_register('autoload_printers');
すべて問題なく動作しているように見えますが、これが実際に許容できるプラクティスと見なされていることを再確認したいと思います。
確かに、よさそうだ。あなたがするかもしれない唯一のことは、彼らがヒットする可能性が最も高い順にそれらを登録することです。たとえば、最も一般的に使用されるクラスがサービス、次にvos、次にプリンターの場合、現在の順序は完璧です。これは、キューに入れられて順番に呼び出されるため、これを行うことでパフォーマンスが少し向上します。
あなたは使うことができます:
set_include_path(implode(PATH_SEPARATOR, array(get_include_path(), './services', './vos', './printers')));
spl_autoload_register();
引数なしで spl_autoload_register
を使用すると、 spl_autoload
が登録されます include path
のディレクトリでクラス名が検索されます 。これは、ファイルシステムで検索する前にクラス名を小文字にすることに注意してください。
大丈夫ですが、これらが特定のフォルダの下にあるフォルダの場合は、たとえば.
/library
/JonoB
/services
/vos
/printers
これらをクラス名に追加することを検討してください。
JonoB_Services_Foo, JonoB_Vos_Bar, JonoB_Printers_Baz
次に、$classname
アンダースコアで、各部分をフォルダ名として使用します。これは PEARクラス名の規則 に似ています。この方法では、ローダーは1つだけになります。
PEAR規則スタイルのクラス名の代わりに、 namespaces ( autoload example )を使用することもできますが、これらにはPHP5.3が必要であることに注意してください。共有ホスティングではまだ広く利用できませんが、アプリケーションはPHP <5.3と下位互換性がありません(それが問題である場合)。
他のすべての回答からの良いアドバイス。
各オートローダーは、渡されるクラスについても気にするかどうかを最初にチェックし、そうでない場合はすぐに戻る必要があることを付け加えておきます。
したがって、Gordonが提案するようにして各クラスにプレフィックスを追加すると、_Services_Foo
_の場合、オートローダーautoload_services()
は、「Services_」が_$class_name
_の最初のサブストリングであるかどうかを確認し、以降の処理、特にファイルシステムのチェックを保存するためにすぐにfalseを返さないでください。
もし私がrefactor your code
、それは
function custom_autoload($class_name){
$dirs = array('services','vos','printers')
foreach($dirs as $dir){
$file = $dir.'/'.$class_name. '.php';
if (file_exists($file)){
require $file;
break; // if i have maintained naming conventions as per dir as there
// is no point for looking eg: sample1_printer.php in the vos/
// or printer/. this way iam avoiding unnecessary loop
}
}
}
spl_autoload_register('custom_autoload');
Spl_autoload_registerを使用して独自のClassLoaderを記述しました。
利点は、関数が現在のフォルダーから始まるすべてのサブフォルダーを検索することです。
私はこのファイルをすべてのPHPファイルに含めるだけで、include/requireディレクティブについて心配する必要はありません。
それは単に機能します:-)
<?php
spl_autoload_register('AutoLoadClasses');
/************************************************************************************
* AutoLoadClasses
*
* Diese Funktion lädt Klassen in gleichnamigen Dateien bei Bedarf automatisch nach,
* sobald eine (bis dahin unbekannte) Klasse erstmalig instanziert wird.
* $var = new MeineKlasse; => Es wird nach der Datei class_MeineKlasse.php gesucht
* Die Suche erfolgt rekursiv in allen Unterordnern ausgehend von dem Ordner, in dem
* das aufrufende PHP-Script liegt.
*
* Michael Hutter / Dezember 2017
*/
function AutoLoadClasses($Klassenname, $StartOrdner = null)
{
if (is_null($StartOrdner))
{
$StartOrdner = __DIR__; # Ausgangspunkt für die Suche: Ordner, in dem sich das aufrufende PHP-Script befindet
$StartInstanz = true;
}
$ZielDateiname = "class_$Klassenname.php";
$FileList = scandir($StartOrdner, 1); # Sortierung 1 => kommt schneller zum Ziel, falls Ordnernamen im allgemeinen mit einem Großbuchstaben beginnen
foreach ($FileList as $file) # Alle Dateien und Ordner durchgehen
{
$Vollpfad = $StartOrdner.DIRECTORY_SEPARATOR.$file;
if (is_dir($Vollpfad) && (substr($file, 0, 1) !== '.')) # Ordner?
{
#echo "Ordner $StartOrdner<br>";
$result = AutoLoadClasses($Klassenname, $Vollpfad);
if ($result) return; # Abbruch, falls Ziel gefunden
}
else if (preg_match('/\.php$/i' , $file)) # .php-Datei?
{
#echo "$file<br>";
if ($file == $ZielDateiname) # Dateiname entspricht Klassenname?
{
include $Vollpfad;
return true; # Abbruch aller Rekursionen, da Ziel gefunden
}
}
}
if (isset($StartInstanz))
die("<table border bgcolor=red><tr><td>Fehler: Die Datei <b>$ZielDateiname</b> konnte in keinem der Unterordner gefunden werden!</td></tr></table>");
return false;
}
?>