将自定义FileInfo类设置为Iterator的正确方法

时间:2010-04-03 21:16:19

标签: php iterator spl

我正在尝试通过setInfoClass方法将自定义类设置为迭代器:

  

使用此方法设置将在调用getFileInfo和getPathInfo时使用的自定义类。传递给此方法的类名必须从SplFileInfo派生。

我的课就是这样(简化示例):

class MyFileInfo extends SplFileInfo
{
    public $props = array(
        'foo' => '1',
        'bar' => '2'
    );
}

迭代器代码是这样的:

$rit = new RecursiveIteratorIterator(
           new RecursiveDirectoryIterator('/some/file/path/'),
           RecursiveIteratorIterator::SELF_FIRST);

由于RecursiveDirectoryIterator是通过DirectoryIterator继承SplFileInfo对象,因此它提供了setInfoClass方法。它没有在手册中列出,但反射显示它在那里:

shell$ php --rc RecursiveDirectoryIterator
// ...
Method [ <internal:SPL, inherits SplFileInfo> public method setInfoClass ] {  
  - Parameters [1] {
    Parameter #0 [ <optional> $class_name ]
  }
}

所有这些都很好,但是当用

遍历目录时
$rit->getInnerIterator()->setInfoClass('MyFileInfo');
foreach($rit as $file) {
    var_dump( $file );
}

我得到以下奇怪的结果

object(MyFileInfo)#4 (3) {
  ["props"]=>UNKNOWN:0
  ["pathName":"SplFileInfo":private]=>string(49) "/some/file/path/someFile.txt"
  ["fileName":"SplFileInfo":private]=>string(25) "someFile.txt"
}

所以当MyFileInfo被选中时,我无法访问它的属性。如果我添加自定义方法,我可以调用它们,但任何属性都是UNKNOWN。

如果我没有将info类设置为迭代器,而是设置为SplFileInfo对象(如手册中的示例所示),它将给出相同的UNKNOWN结果:

foreach($rit as $file) {
    // $file is a SplFileInfo instance
    $file->setInfoClass('MyFileInfo');
    var_dump( $file->getFileInfo() );
}

然而,当我这样做时它会起作用

foreach($rit as $file) {
    $file = new MyFileInfo($file);
    var_dump( $file );
}

不幸的是,我想要使用它的代码有点复杂,并且堆栈了更多的迭代器。像这样创建MyFileInfo类不是一种选择。

那么,有没有人知道如何使这个工作或为什么PHP表现得这么奇怪?

感谢。

1 个答案:

答案 0 :(得分:2)

不能确切地告诉你为什么,但它适用于

class MyFileInfo extends SplFileInfo
{
  public $props;

  public function __construct($filename) {
    $this->props = array(
      'foo' => '1',
      'bar' => '2'
    );

    parent::__construct($filename);
  }
}