【问题标题】:zend framework2 how does the autoload function workzend framework2的自动加载功能是如何工作的
【发布时间】:2013-10-03 20:16:58
【问题描述】:

最近在学习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'

我尝试更改命名空间的名称,例如 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”付诸实践?我会等待你的答复。

【问题讨论】:

标签: php namespaces zend-framework2 autoloader


【解决方案1】:

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

命名空间允许您或多或少地为您的类创建“目录”。因此,您可以创建 \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';
    }
}

最后,让我们谈谈自动加载。您不需要将 use 与自动加载一起使用。您的自动装载机应该会为您处理好这些。但是,您应该使用spl_autoload_register(),因为__autoload() 很快就会被贬值。

【讨论】:

  • 但是还有一些不清楚的地方,当然我知道在我的程序中没有必要使用命名空间,我想做的是模拟 ZF2 以了解为什么在 ZF2 中我们可以这样编写代码(例如作为我线程中的第一个程序),那么你能解释一下 ZF2 中到底发生了什么,为什么在 ZF2 中使用命名空间和一起使用没有问题,但我的程序不起作用?或者你能给我一些例子吗?非常感谢。
  • 如果您查看 ZF2 中的 public/index.php,您会发现它不使用命名空间。命名空间是一种面向对象的编程原则,正如@Machavity 所说,它允许在逻辑上分离你的类。 PHP,就其本质而言,并不是纯粹的面向对象的。所以与请求交互的第一件事(在这种情况下为 index.php)通常没有命名空间,因为它不是一个类,只是一个实例化自动加载器和应用程序对象的过程文件。
【解决方案2】:

来自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,你会看到..

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2019-01-01
    • 2019-12-16
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-06-11
    相关资源
    最近更新 更多