Zend 框架2 自动加载函数如何工作


zend framework2 how does the autoload function work

最近我在学习 zend Framework 2,有一个问题困扰了我很长时间,事情看起来像这样:

<?php
namespace Album'Model;
// Add these import statements
use Zend'InputFilter'InputFilter;
use Zend'InputFilter'InputFilterAwareInterface;
use Zend'InputFilter'InputFilterInterface;
class Album implements InputFilterAwareInterface
{
    public $id;
    public $artist;
    public $title;
    protected $inputFilter;
    public function exchangeArray($data)
    {
        $this->id     = (isset($data['id']))     ? $data['id']     : null;
        $this->artist = (isset($data['artist'])) ? $data['artist'] : null;
        $this->title  = (isset($data['title']))  ? $data['title']  : null;
    }
    // Add content to these methods:
    public function setInputFilter(InputFilterInterface $inputFilter)
    {
        throw new 'Exception("Not used");
    }
    //....
    ?>

此代码是"骨架应用程序"程序的一部分,该程序是 ZF2 的教程。第一次看到程序的时候,我不明白"namespace"和"use"是什么用法,因为这两个关键字在 php5.2 中不存在(在早期版本中也一样(,所以我去看手册并尝试理解它。我写了一个程序来模拟真实发生的事情:

<?php
use script'lib'test;
$o = new test();
echo $o->getWelcome();
function __autoload( $className ) {  
$classname = strtolower( $classname );  
require_once( dirname( __FILE__ ) . '/' . $classname . '.php' );  
}
?>

上面的程序运行良好,当然我创建了两个名为 script 和 lib 的文件夹,还有一个名为 test.php 的文件。似乎一切都很清楚,zend 框架也有一个自动加载功能,但是当我注意到"骨架应用程序"中的代码时,一开始有一个命名空间,所以我也把命名空间添加到我的程序中:

<?php
namespace test;
use script'lib'test;
$o = new test();
echo $o->getWelcome();
function __autoload( $className ) {  
$classname = strtolower( $classname );  
require_once( dirname( __FILE__ ) . '/' . $classname . '.php' );  
}
?>

该页面向我返回的信息如下:致命错误:在第 6 行的 E:''wamp''www''test''test_29.php 中找不到类"script''lib''test">

我尝试更改命名空间的名称,例如脚本''库、脚本''库''测试...但没用。

任何答案将不胜感激,谢谢。


现在,我将为您提供有关此问题的更多详细信息:为了理解"命名空间"和"use"的用法,我查看了 php.net 上的材料:http://php.net/manual/en/language.namespaces.importing.php在此页面中,有一段代码如下所示:

示例 #1 使用 use 运算符导入/别名

<?php
namespace foo;
use My'Full'Classname as Another;
// this is the same as use My'Full'NSname as NSname
use My'Full'NSname;
// importing a global class
use ArrayObject;
$obj = new namespace'Another; // instantiates object of class foo'Another
$obj = new Another; // instantiates object of class My'Full'Classname
NSname'subns'func(); // calls function My'Full'NSname'subns'func
$a = new ArrayObject(array(1)); // instantiates object of class ArrayObject
// without the "use ArrayObject" we would instantiate an object of class
?>

现在让我们回顾一下我在上面写的程序:

<?php
namespace test;
use script'lib'test;
$o = new test();
echo $o->getWelcome();
function __autoload( $className ) {  
$classname = strtolower( $classname );  
require_once( dirname( __FILE__ ) . '/' . $classname . '.php' );  
}
?>

是一样的,如果我们不使用自动加载功能,我正在尝试模拟该实例:

<?php
namespace test;
use script'lib'test;
require_once 'script/lib/test.php';
$o = new test();
echo $o->getWelcome();
?>

它也运行良好,但是当我使用__autoload函数加载类文件时,出现了问题。我不知道问题出在哪里,或者任何机构试图编写一个实例来将"示例#1"付诸实践?我会等待你的回答。

我想

你误解了这里发生的事情。

命名空间允许您或多或少地为类创建"目录"。因此,您可以创建 ''Foo 类和 ''Test''Foo 类(其中 '' 表示应用程序的"根"(。自动加载的工作方式是文件镜像您的命名空间。所以 foo.php 将在自动加载的根目录中,但您将为 ''Test''Foo 创建/test/foo.phpuse 关键字有两种用途。一种是别名类文件,另一种是在 PHP 5.4 或更高版本中将 Trait 引入当前类。

现在,回答你的问题。首先,让我们看一下你的代码

<?php
namespace test;
use script'lib'test;
$o = new test();
echo $o->getWelcome();

这是令人困惑的。你声明一个命名空间(你不需要在这里做(,但你把它别名为脚本''lib''test。PHP 现在正在寻找一个名为/script/lib/test.php 的文件,您的错误消息说该文件不存在。但是你说文件确实存在,所以让我们看看

public function getWelcome() {
     return 'welcome';
}

这不是一个类。这是一个函数。对于此示例,您需要一个完整的类

<?php
namespace script'lib;
class test {
    public function getWelcome() {
        return 'welcome';
    }
}

最后,让我们谈谈自动加载。您无需将 use 与自动加载一起使用。您的自动加载机应该为您处理这个问题。但是,您应该使用 spl_autoload_register((,因为 __autoload(( 很快就会被折旧。

来自 ZF2 docu

Zend''Loader''StandardAutoloader 被设计为符合 PSR-0 标准的自动加载机。它假定命名空间+类名与文件系统的 1:1 映射,其中命名空间分隔符和下划线转换为目录分隔符。

阅读更多关于: PSR-0

因此,如果您使用命名空间,则发送到自动加载器的类名看起来不像test。看起来像YOUR_NAMESPACE'test.YOUR_NAMESPACE 是在类中定义的命名空间,其中包含 namespace YOUR_NAMESPACE;

PSR-0 是一个标准,它说:你的命名空间应该反映你的文件系统。您只需将反斜杠替换为正斜杠。或者,如果您使用的是 ZF1 中的伪命名空间,请使用 / _。(Album_Model_Album(

因此,输出发送到自动加载机的$className,您将看到..