我正在学习使用php autuoloader
...
据我所知,我们可以使用__autoloader
或spl_autoloader_*
自动加载文件。
假设这是我的目录结构:
ROOT
|
|
ADMIN
| |
| |
| DIST
| |
| SOME_FOLDER
| SOME_FOLDER
| TPL
| |
| |
| SOME_FOLDER1
| |
| test.php
| SOME_FOLDER2
| |
| example1.php
| example2.php
| example3.php
|
|
CLASSES
|
basics.php
class1.php
class2.php
class3.php
class4.php
|
index.php
我在CLASSES
目录中为自动加载文件创建了这个类:
basics.php:
class MyAutoLoader
{
public function __construct()
{
spl_autoload_register( array($this, 'load') );
}
function load( $file )
{
//spl_autoload( 'load1' );
echo 'Try to call ' . $file . '.php inside ' . __METHOD__ . '<br>';
require( $file . '.php' );
}
}
并在index.php
我将包含basics.php
,并且每件事都可以将文件存储在CLASSES
文件夹中...
require_once("CLASSES/basics.php");
$loaderObject = new MyAutoLoader();
使用此代码,我可以声明class1
...
class3
现在我想要一个可以在SOME_FOLDER2
加载文件的自动加载器,在这种情况下是example1.php
,example2.php
和example3.php
我尝试了一些案例,但SOME_FOLDER2
中的文件不会使用自动加载器加载。
我的尝试:
我在load2
类中创建了一个名为MyAutoLoader
的函数,该函数尝试包含SOME_FOLDER2
function load2( $file )
{
//spl_autoload_register('load2');
echo 'Inside LOADER2 ' . __METHOD__ . '<br>';
require ( 'ADMIN/TPL/' . $file . '.php' );
}
我在spl_autoload_register
构造函数中更改了MyAutoLoader
:
$allMethods = get_class_methods( 'MyAutoLoader' );
$allMethods = array_splice( $allMethods, 1 );
foreach( $allMethods as $method )
{
spl_autoload_register( array($this, $method) );
}
但他们都没有为我工作......
你能告诉我我的代码中有什么问题吗?或者我对auloader的误解是什么?
的 Thanks in Advance
答案 0 :(得分:7)
我认为你的问题基本上归结为在require
之前没有检查文件是否存在...如果你试图包含的第一个文件夹中不存在该文件会产生致命错误从
我不知道你的用例,但这里有一些建议:
你能使用一个自动加载器吗?
function my_autoload($class_name) {
if (is_file('CLASSES/' . $class_name . '.php')) {
require_once 'CLASSES/' . $class_name . '.php';
} else if (is_file('ADMIN/TPL/SOME_FOLDER2/' . $class_name . '.php')) {
require_once 'ADMIN/TPL/SOME_FOLDER2/' . $class_name . '.php';
}
}
spl_autoload_register("my_autoload");
或者如果您需要独立声明它们(即两个或更多个自动加载器):
function classes_autoload($class_name) {
if (is_file('CLASSES/' . $class_name . '.php')) {
require_once 'CLASSES/' . $class_name . '.php';
}
}
spl_autoload_register("classes_autoload");
function admin_autoload($class_name) {
if (is_file('ADMIN/TPL/SOME_FOLDER2/' . $class_name . '.php')) {
require_once 'ADMIN/TPL/SOME_FOLDER2/' . $class_name . '.php';
}
}
spl_autoload_register("admin_autoload");
或者使您的自动加载器类具有通用性:
class MyAutoLoader {
private $path;
public function __construct($path) {
$this->path = $path;
spl_autoload_register( array($this, 'load') );
}
function load( $file ) {
if (is_file($this->path . '/' . $file . '.php')) {
require_once( $this->path . '/' . $file . '.php' );
}
}
}
$autoloader_classes = new MyAutoLoader('CLASSES');
$autoloader_admin = new MyAutoLoader('ADMIN/TPL/SOME_FOLDER2');
如果你不想手动保持ADMIN/TPL
内的子文件夹列表是最新的,你甚至可以做这样的事情从其中任何一个自动加载(这显然假设{{1}的所有子文件夹包含类):
ADMIN/TPL
答案 1 :(得分:1)
您的问题实际上可能就像这样简单:
您的basics.php
位于文件夹CLASSES
内,因此当include
或require
内的文件从basics.php
开始时,它就会在该文件夹中启动。
因此,如果你想要包含来自ADMIN/TPL/SOME_FOLDER2
的文件,你实际上需要首先“跳”到一个级别,即require '../ADMIN/TPL/SOME_FOLDER2/' . $file . '.php';
为了避免这种混淆,我建议在index.php
的开头添加一个常量,如下所示:
define('BASEPATH', __DIR__);
...然后将其添加到所有require
和include
语句中 - 即:
class MyAutoLoader {
public function __construct() {
spl_autoload_register( array($this, 'load') );
spl_autoload_register( array($this, 'load2') );
}
function load( $file ) {
echo 'Try to call ' . $file . '.php inside ' . __METHOD__ . '<br>';
$classfile = BASEPATH . '/CLASSES/' . $file . '.php';
if ( is_file( $classfile ) ) {
require( $classfile );
}
}
function load2( $file ) {
echo 'Try to call ' . $file . '.php inside ' . __METHOD__ . '<br>';
$classfile = BASEPATH . '/ADMIN/TPL/SOME_FOLDER2/' . $file . '.php';
if ( is_file( $classfile ) ) {
require( $classfile );
}
}
}