从邻接列表数据中获取路径

时间:2015-06-25 12:07:39

标签: php arrays adjacency-list

我有一个数组(来自邻接表的数据),它看起来像:

Array
(
    [0] => Array
        (
            [id] => 1
            [name] => Anniversary
            [parent] => 0
        )

    [1] => Array
        (
            [id] => 12
            [name] => New arrives
            [parent] => 1
        )

    [2] => Array
        (
            [id] => 13
            [name] => Discount
            [parent] => 12
        )

    [3] => Array
        (
            [id] => 6
            [name] => Birthday
            [parent] => 0
        )
)

我正在寻找通过ID检索路径的方法;

For example: getPath(13): Anniversary->New arrives->Discount;
For example: getPath(12): Anniversary->New arrives;
For example: getPath(1): Anniversary;
For example: getPath(6): Birthday;

我该怎么做? 谢谢!

3 个答案:

答案 0 :(得分:2)

 a = a - 2 * b < 0 ? 0 : a;

答案 1 :(得分:2)

考虑这个数组,

$input = [
    ['id'=>1, 'name'=>'Anniversary', 'parent'=>0],
    ['id'=>12, 'name'=>'New arrives', 'parent'=>1],
    ['id'=>13, 'name'=>'Discount', 'parent'=>12],
    ['id'=>6, 'name'=>'Birthday', 'parent'=>0]
];

和这个功能,

function path($element_id, $input, $ids = [])
{
    if(!$ids) // for performance, make this once and pass it around
    {   
        $ids = array_column($input, 'id'); // array containing only 'id's of $input
    }   

    $current_key = array_search($element_id, $ids); // search for $input variable's current key

    unset($ids[$current_key]); // unsetting used keys to make above array search faster next time

    $current_element = $input[$current_key]; // get current element as array from $input

    $names[] = $current_element['name']; // create an array containing current element

    if($current_element['parent'] != 0) // check if current element have parent
    {   
        $names[] = path($current_element['parent'], $input, $ids); // call this function, let it return string, append it to $names
    }   

    return implode(' ⟶ ', array_reverse($names)); // make final return, seprate by ⟶
}

阅读echo path(13, $input);将返回

Anniversary ⟶ New arrives ⟶ Discount

这是相同功能的缩小版

function path($a,$b,$c=[]){if(!$c){$c=array_column($b,'id');}$d=array_search($a,$c);unset($c[$d]);$e=$b[$d];$f[]=$e['name'];if($e['parent']!=0){$f[]=path($e['parent'],$b,$c);}return implode(' ⟶ ',array_reverse($f));}

感谢代码审核人LoufyloufQuill

答案 2 :(得分:1)

$find = 13;
$path = array();

function FindById ($arr, $find) {
  $k = null;

  foreach($arr as $key => $item) 
    if ($item['id'] == $find) 
      { $k = $key; break; }
  return $k;
}

if ( false === ($k = FindById($arr, $find)))  die("not found");

while (true) {
   array_unshift($path, $arr[$k]['name']);
   if( !  $arr[$k]['parent']) break;
   if(false === ($k = FindById($arr, $arr[$k]['parent']))) die("illegal structure");

}  

echo implode('->', $path);