使用foreach循环递归

时间:2013-10-20 20:49:55

标签: php recursion

我正在尝试构建一个递归函数,但在某些方面函数不会显示所有目录。我错过了什么?我知道我可以使用for或while,但我想要foreach循环。这是代码:

$GLOBALS['testmodus']=0;
$o=new BrowseFolder();
$o->browse('../images/',1);
ta($o);
function ta($ausgabe) {
  echo('<p class="ta">'.__LINE__.$ausgabe.'</p>');
 }


class BrowseFolder{
    private $srcFolder;
    private $ifRecursiv=0;
    private $excludeFileTypes=array('.','..');
    private $filesFound;
    private $deleteFolderPath=0;
    private $makeFolderPath;

    public function browse($srcFolder,$ifRecursiv=0){
        ta($this->filesFound);
        $this->srcFolder=$srcFolder;
        $this->ifRecursiv=$ifRecursiv;
        $this->filesFound=scandir($this->srcFolder);
        $this->filesFound=$this->excludeArrayFromArray($this->filesFound,$this->excludeFileTypes);
        echo "<ul>";
        foreach($this->filesFound as $val){
                if(is_file($this->srcFolder.$val)){
                    echo "<li>";
                    echo "$val";
                    echo "</li>";
                    }elseif(is_dir($this->srcFolder.$val) && $this->ifRecursiv==1){
                        ta($this->srcFolder.$val);
                        echo "$val";
                        $this->browse($this->srcFolder.$val.'/',$this->ifRecursiv);

                      }
        }
        echo "</ul>";
    }

    private function excludeArrayFromArray($baseArray,$arrayToExclude){  //Exclude an array from another array,and arrange
        $newArray=array_values(array_diff($baseArray,$arrayToExclude));  
        return $newArray;

    }
}

1 个答案:

答案 0 :(得分:1)

我猜想如果你查看网络服务器日志,你会看到这个错误:

PHP Catchable fatal error:  Object of class BrowseFolder could not be converted to string in <filename> on line <XX>

您的BrowseFolder类没有__toString()方法,因此尝试在字符串上下文中调用它会导致致命错误(此行:ta($o);

然而,这是唯一的问题1。 问题2是这样的:

$this->srcFolder=$srcFolder;

由于BrowseFolder类只有一个实例,因此您在类上设置属性,然后尝试在递归中重用该属性。所以基本上它正在做的是钻进第一个文件夹,然后因为$this->srcFolder只等于后续的孩子,它不会进一步尝试。仅使用$this->srcFolder替换$srcFolder的每个实例。递归函数将向下钻取到第一个函数,然后当它到达结束时,返回到browse()的第一个调用,然后继续到第二个调用,依此类推。

修改 - 其他详细信息: 每次都不应该修改变量。它正在做的是获取第一个文件夹images,然后在类上设置该属性。它会进入该文件夹的第一个子节点 - 称之为folder1,然后再将类属性设置为该子节点(即$this->srcFolder == "images/folder1")。它列出该文件夹中的文件,然后继续。由于该文件夹中没有更多文件或子目录,因此它已到达该迭代的末尾。然后它返回到images文件夹的级别。它转到foreach循环的下一次迭代,现在将$val设置为folder2。现在它同时尝试if(is_file($this->srcFolder.$val))else if (is_dir($this->srcFolder.$val)。由于is_file("images/folder1/folder2")等于$this->srcFolder,因此images/folder1的翻译为false。因此,它始终会返回foreach,直到它到达$srcFolder的末尾并退出。

如果你改为调用folder1(局部变量),那么每次进入函数时它都会有正确的值。在同一场景中:它进入browse()并进入$srcFolder == "images/folder1"函数的第二级images。它列出了该文件夹的每个元素,然后返回到$srcFolder。现在,该函数仍然具有"images"的初始值,即foreach。那么在$val循环的第二次迭代中,现在folder2再次等于browse(),因此它会调用$srcFolder == "images/folder2"。再次在函数的第二级{{1}}。在递归中使用局部变量可确保函数的每次调用都具有所需的值。