如何递归检查包含其他依赖项的文件包含依赖项的数组?
为了解释我的问题,让我先点这个数组:如何递归检查包含其他依赖项的文件包含依赖项的数组?
<?php
$_depends = array(
'/Scripting/jquery.hyponiqs/jquery.dropdown.js' => array(
"/Scripting/jquery.externals/jquery.resize.js",
"/Scripting/jquery.externals/jquery.topzindex.js",
"/Scripting/jquery.externals/jquery.timers.js",
"/Scripting/jquery-ui.min.js",
"/Scripting/jquery.hyponiqs/hyponiqs.core.js",
),
'/Script/UI/Dialogs.js' => array(
"/Scripting/jquery.externals/jquery.resize.js",
"/Scripting/jquery.externals/jquery.topzindex.js"
),
'/Script/Display/List.js' => array(
"/Scripting/jquery.externals/jquery.timers.js"
)
);
?>
每当包含的JavaScript文件,它是对这个数组的依赖性检查。然后将每个文件的所有依赖关系添加到最终的$includes
阵列中。问题是当我添加具有依赖性的包括那些依赖的一个也有其自己的依赖,比如:
<?php
$_depends = array(
'/Scripting/jquery.hyponiqs/jquery.dropdown.js' => array(
"/Scripting/jquery.externals/jquery.resize.js",
"/Scripting/jquery.externals/jquery.topzindex.js",
"/Scripting/jquery.externals/jquery.timers.js",
"/Scripting/jquery-ui.min.js",
"/Scripting/jquery.hyponiqs/hyponiqs.core.js",
),
'/Script/UI/Dialogs.js' => array(
"/Scripting/jquery.externals/jquery.resize.js",
"/Scripting/jquery.externals/jquery.topzindex.js"
),
'/Script/Display/List.js' => array(
"/Scripting/jquery.externals/jquery.timers.js"
),
'/Script/UI/Calendar/Main.js' => array(
"/Scripting/jquery-ui.min.js",
"/Script/UI/Dialogs.js"
)
);
?>
正如你所看到的,所添加的'/Script/UI/Calendar/Main.js'
取决于"/Script/UI/Dialogs.js"
有其自己的依赖。
我知道我将不得不递归检查依赖数组和final包含数组,但我似乎无法将我的头围绕逻辑。这里有一点帮助可能会很好。
UPDATE
我在一类包裹的一切来说明它的目的(尽管实际的类要复杂得多,有各种其他包括处理功能:
<?php
class Script_Depends {
private $_includes = array();
private $_depends = array(
'/Scripting/jquery.hyponiqs/jquery.dropdown.js' => array(
"/Scripting/jquery.externals/jquery.resize.js",
"/Scripting/jquery.externals/jquery.topzindex.js",
"/Scripting/jquery.externals/jquery.timers.js",
"/Scripting/jquery-ui.min.js",
"/Scripting/jquery.hyponiqs/hyponiqs.core.js",
),
'/Script/UI/Dialogs.js' => array(
"/Scripting/jquery.externals/jquery.resize.js",
"/Scripting/jquery.externals/jquery.topzindex.js"
),
'/Script/Display/List.js' => array(
"/Scripting/jquery.externals/jquery.timers.js"
),
'/Script/UI/Calendar/Main.js' => array(
"/Script/UI/Dialogs.js",
"/Scripting/jquery-ui.min.js"
)
);
public function includes($includes)
{
if (is_string($includes)) $includes = array($includes);
foreach ($includes as $include) {
if (isset($this->_depends[$include])) {
$this->_includes = $this->includes($this->_depends[$include]);
array_push($this->_includes, $include);
}
else {
array_push($this->_includes, $include);
}
}
$this->_includes = array_unique($this->_includes);
return $this->_includes;
}
}
?>
你正在寻找被称为扁平化功能。它需要递归列表并返回每个级别的所有元素。
这里有一些在PHP实现的评论:http://php.net/manual/en/function.array-values.php
这是在JavaScript中实现:http://tech.karbassi.com/2009/12/17/pure-javascript-flatten-array/
不,这不是一个flatenning。展平是指具有多维数组并将一个或多个层级折叠到基层。 – Artefacto 2010-08-10 13:02:59
也许我应该进一步解释一下:并非所有这些都将被包含在所有的时间,只有一些时间取决于所要求的。 I.E.我可能只在一个应用程序中包含'/Script/Display/List.js',而在另一个应用程序中包含'/Script/UI/Calendar/Main.js'和'/Scripting/jquery.hyponiqs/jquery.dropdown.js'。 – hyponiq 2010-08-10 13:05:02
你是对的;我读得太快了。这是图中边的列表;根本问题是从一组起始节点中查找图表中可到达的节点。 – Cirdec 2010-08-11 02:29:36
试试这个:
function getDependentFiles($file, $dependencies) {
$fileIndex = array();
$queue = array($file);
while (count($queue)) {
$f = array_shift($queue);
if (!isset($fileIndex[$f])) {
if (array_key_exists($f, $dependencies) && is_array($dependencies[$f])) {
$queue = array_merge($queue, $dependencies[$f]);
}
$fileIndex[$f] = true;
}
}
return array_keys($fileIndex);
}
var_dump(getDependentFiles('/Script/UI/Calendar/Main.js', $_depends));
这基本上实现了由Artefacto描述的算法。
谢谢...但我已经实现了答案,我有这个时间。并且,感谢你们两位的帮助! – hyponiq 2010-08-10 13:13:44
看看这是否有窍门 - 对你的功能的一个小型修改。(顺便说一句,因为你的命名约定:$ includes,$ include,$ this - > _ includes和$ this-> includes()!!建议你使用更好的名字。我已经重命名了函数以calculate_includes,但你应该清理其他人。)
public function calculate_includes($includes) {
foreach ((array)$includes as $include) {
if(in_array($include, $this->_includes)) continue;
if (isset($this->_depends[$include])) {
$this->_includes = $this->calculate_includes($this->_depends[$include]);
} else {
array_push($this->_includes, $include);
}
}
$this->_includes = array_unique($this->_includes);
return $this->_includes;
}
感谢您的帮助。我无法相信答案是那么简单......而且我没有看到它! – hyponiq 2010-08-10 13:11:43