php 無(wú)限級(jí)分類學(xué)習(xí)參考之對(duì)ecshop無(wú)限級(jí)分類的解析 帶詳細(xì)注釋
更新時(shí)間:2010年03月23日 14:17:56 作者:
對(duì)ecshop無(wú)限級(jí)分類的解析,認(rèn)真分析后發(fā)現(xiàn)真的其算法還是比較精典的其實(shí)并不難理解,有舉例方便大家理解。
復(fù)制代碼 代碼如下:
function cat_options($spec_cat_id, $arr)
{
static $cat_options = array();
if (isset($cat_options[$spec_cat_id]))
{
return $cat_options[$spec_cat_id];
}
/*
初始化關(guān)鍵參數(shù):
$level:當(dāng)前子節(jié)點(diǎn)深度
$last_cat_id:當(dāng)前父節(jié)點(diǎn)ID
$options:帶有縮進(jìn)級(jí)別的數(shù)組
$cat_id_array:沿同一路徑的父節(jié)點(diǎn)依次進(jìn)駐
$level_array:該節(jié)點(diǎn)的子節(jié)點(diǎn)深度,也是依次進(jìn)駐
*/
if (!isset($cat_options[0]))
{
$level = $last_cat_id = 0;
$options = $cat_id_array = $level_array = array();
while (!empty($arr))//如果還有待構(gòu)造的節(jié)點(diǎn)則繼續(xù)遍歷
{
foreach ($arr AS $key => $value)
{
$cat_id = $value['cat_id'];
//一級(jí)分類結(jié)點(diǎn)
if ($level == 0 && $last_cat_id == 0)
{
if ($value['parent_id'] > 0)
{
break;
}
$options[$cat_id] = $value;
$options[$cat_id]['level'] = $level;
$options[$cat_id]['id'] = $cat_id;
$options[$cat_id]['name'] = $value['cat_name'];
//遍歷過(guò)了就不再遍歷
unset($arr[$key]);
if ($value['has_children'] == 0)
{
continue;
}
$last_cat_id = $cat_id;//下層結(jié)點(diǎn)的父親結(jié)點(diǎn)
$cat_id_array = array($cat_id);
$level_array[$last_cat_id] = ++$level;
continue;
}
//當(dāng)前結(jié)點(diǎn)的父親結(jié)點(diǎn)ID等于它的上一級(jí)結(jié)點(diǎn)ID
if ($value['parent_id'] == $last_cat_id)
{
$options[$cat_id] = $value;
$options[$cat_id]['level'] = $level;
$options[$cat_id]['id'] = $cat_id;
$options[$cat_id]['name'] = $value['cat_name'];
unset($arr[$key]);//遍歷過(guò)了就不再遍歷
//如果當(dāng)前結(jié)點(diǎn)有孩子則當(dāng)前結(jié)點(diǎn)要進(jìn)駐,但不再遍歷;反之不進(jìn)駐也不再遍歷
if ($value['has_children'] > 0)
{
if (end($cat_id_array) != $last_cat_id)
{
$cat_id_array[] = $last_cat_id;
}
$last_cat_id = $cat_id;//當(dāng)現(xiàn)結(jié)點(diǎn)做為下一級(jí)結(jié)點(diǎn)的新的父親結(jié)點(diǎn)
$cat_id_array[] = $cat_id;//進(jìn)駐
$level_array[$last_cat_id] = ++$level;//當(dāng)前結(jié)點(diǎn)的下一級(jí)結(jié)點(diǎn)深度
}
}
elseif ($value['parent_id'] > $last_cat_id)
{//如果當(dāng)前結(jié)點(diǎn)父親深度大于目前父親結(jié)點(diǎn)的深度則進(jìn)行下一輪循環(huán)
break;
}
}//endforeach
$count = count($cat_id_array);
if ($count > 1)
{
//取出最后進(jìn)駐的父親節(jié)點(diǎn)作為當(dāng)前父親節(jié)點(diǎn)
$last_cat_id = array_pop($cat_id_array);
}
elseif ($count == 1)
{
if ($last_cat_id != end($cat_id_array))
{
//進(jìn)駐的父親結(jié)點(diǎn)只有一個(gè)時(shí)并且沒(méi)有作為當(dāng)前父親節(jié)點(diǎn)時(shí)把它取出
$last_cat_id = end($cat_id_array);
}
else
{ //否則最后取出的父親結(jié)點(diǎn)一定是一級(jí)分類結(jié)點(diǎn)
$level = 0;
$last_cat_id = 0;
$cat_id_array = array();
continue;
}
}
if ($last_cat_id && isset($level_array[$last_cat_id]))
{
//取出當(dāng)前結(jié)點(diǎn)的深度
$level = $level_array[$last_cat_id];
}
else
{
$level = 0;
}
}//end while,此時(shí)已完成非遞歸前序遍歷構(gòu)造樹的工作,其中$options已保存了從根結(jié)點(diǎn)開始的所有結(jié)點(diǎn)帶有分層性質(zhì)的數(shù)組
$cat_options[0] = $options;
}
else
{
$options = $cat_options[0];
}
//如果從0開始即取整個(gè)樹則直接返回不再處理.
if (!$spec_cat_id)
{
return $options;
}
//否則開始從指定結(jié)點(diǎn)截取,以下比較簡(jiǎn)單我還是稍微說(shuō)說(shuō)吧,要說(shuō)就說(shuō)幾個(gè)參數(shù)含義吧
/*
$spec_cat_id_level:截取結(jié)點(diǎn)的深度
$spec_cat_id_array:最終返回的以該結(jié)點(diǎn)為根結(jié)點(diǎn)的一棵商品分類樹
最終返回的數(shù)組是這樣排序的:按父親結(jié)點(diǎn)大小,按直接父親結(jié)點(diǎn),按同一父親結(jié)點(diǎn)這樣的先根遍歷,具個(gè)例子:
一級(jí)結(jié)點(diǎn)有1,5 二級(jí)結(jié)點(diǎn)有2,6,7 三級(jí)結(jié)點(diǎn)有8,9,如果1的直接孩子是2,6而2的直接孩子是8,9;另外
5的直接孩子是7那么最終的數(shù)組是這樣排列的1->2->8->9->6->5->7
*/
else
{
if (empty($options[$spec_cat_id]))
{
return array();
}
$spec_cat_id_level = $options[$spec_cat_id]['level'];
foreach ($options AS $key => $value)
{
if ($key != $spec_cat_id)
{
unset($options[$key]);
}
else
{
break;
}
}
$spec_cat_id_array = array();
foreach ($options AS $key => $value)
{
if (($spec_cat_id_level == $value['level'] && $value['cat_id'] != $spec_cat_id) ||
($spec_cat_id_level > $value['level']))
{
break;
}
else
{
$spec_cat_id_array[$key] = $value;
}
}
$cat_options[$spec_cat_id] = $spec_cat_id_array;
return $spec_cat_id_array;
}
}
您可能感興趣的文章:
- php實(shí)現(xiàn)無(wú)限級(jí)分類(遞歸方法)
- php 無(wú)限級(jí)分類,超級(jí)簡(jiǎn)單的無(wú)限級(jí)分類,支持輸出樹狀圖
- php實(shí)現(xiàn)無(wú)限級(jí)分類
- php實(shí)現(xiàn)無(wú)限級(jí)分類查詢(遞歸、非遞歸)
- ThinkPHP無(wú)限級(jí)分類原理實(shí)現(xiàn)留言與回復(fù)功能實(shí)例
- php+mysql實(shí)現(xiàn)無(wú)限級(jí)分類 | 樹型顯示分類關(guān)系
- ThinkPHP自動(dòng)填充實(shí)現(xiàn)無(wú)限級(jí)分類的方法
- php實(shí)現(xiàn)無(wú)限級(jí)分類實(shí)現(xiàn)代碼(遞歸方法)
- PHP實(shí)現(xiàn)遞歸無(wú)限級(jí)分類
- PHPTree——php快速生成無(wú)限級(jí)分類
相關(guān)文章
PHP static局部靜態(tài)變量和全局靜態(tài)變量總結(jié)
使用靜態(tài)局部變量的原因是它不能被外部使用,但是它的值依然在沒(méi)此使用后保留。雖然使用全局變量可以實(shí)現(xiàn)相同的功能,但是它經(jīng)常會(huì)造成意外2014-03-03php計(jì)算函數(shù)執(zhí)行時(shí)間的方法
這篇文章主要介紹了php計(jì)算函數(shù)執(zhí)行時(shí)間的方法,以md5函數(shù)加密運(yùn)行時(shí)間為例分析了php計(jì)算函數(shù)運(yùn)行時(shí)間的技巧,需要的朋友可以參考下2015-03-03PHP實(shí)現(xiàn)SQL語(yǔ)句格式化功能的方法
這篇文章主要介紹了PHP實(shí)現(xiàn)SQL語(yǔ)句格式化功能的方法,基于github上開源代碼實(shí)現(xiàn)的SQL語(yǔ)句格式化功能,非常簡(jiǎn)單實(shí)用,需要的朋友可以參考下2017-07-07php判斷上傳的Excel文件中是否有圖片及PHPExcel庫(kù)認(rèn)識(shí)
php判斷Excel文件中是否有圖片,大家很想知道如何實(shí)現(xiàn)吧,不要走開接下來(lái)為您揭曉,感興趣的朋友可以了解下哦2013-01-01php json轉(zhuǎn)換成數(shù)組形式代碼分享
這篇文章主要介紹了php json轉(zhuǎn)換成數(shù)組形式代碼分享,需要的朋友可以參考下2014-11-11