zend framework2自动加载功能如何工作

时间:2013-09-27 13:43:07

标签: php namespaces zend-framework2 autoloader

最近我正在学习zend框架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的教程。我第一次看到程序时,我不明白“命名空间”和“使用”的用法是什么,因为这两个关键字在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'

我尝试更改名称空间的名称,例如script \ lib,script \ lib \ test ... 但它没用。

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


现在我将为您提供有关此问题的更多详细信息: 为了理解“命名空间”和“使用”的用法,我查看了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”?我会等你的回答。

2 个答案:

答案 0 :(得分:4)

我认为你误解了这里发生了什么。

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

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

<?php
namespace test;
use script\lib\test;

$o = new test();
echo $o->getWelcome();

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

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

这不是一个班级。这是一个功能。对于此示例,您需要一个完整的类

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

最后,让我们谈谈自动加载。您无需在自动加载中使用使用。您的自动加载器应该为您处理。但是,您应该使用spl_autoload_register(),因为__autoload()很快就会被弃用。

答案 1 :(得分:1)

来自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,您将看到..