从多维数组创建嵌套列表

时间:2012-10-07 18:44:20

标签: php arrays

我在PHP中有一个数组,如下所示:

array (
    [0] => array (
        [id] => 1
        [title] => "Title 1"
        [parent_id] => NULL
        [depth] => 0
    )
    [1] => array (
        [id] => 2
        [title] => "Title 2"
        [parent_id] => NULL
        [depth] => 0
    )
    [2] => array (
        [id] => 3
        [title] => "Title 3"
        [parent_id] => 2
        [depth] => 1
    )
    [3] => array (
        [id] => 4
        [title] => "Title 4"
        [parent_id] => 2
        [depth] => 1
    )
    [4] => array (
        [id] => 5
        [title] => "Title 5"
        [parent_id] => NULL
        [depth] => 0
    )
    [5] => array (
        [id] => 6
        [title] => "Title 6"
        [parent_id] => 4
        [depth] => 2
    )
)

我想要做的是迭代这个数组并从中创建一个嵌套的<ol>列表。所以结果应该是这样的:

<ol>
    <li>Title 1</li> // id = 1
    <li>Title 2</li> // id = 2
    <ol>
        <li>Title 3</li> // id = 3 -> parent_id = 2
        <li>Title 4</li> // id = 4 -> parent_id = 2
        <ol>
            <li>Title 6</li> // id = 6 -> parent_id = 4
        </ol>
    </ol>
    <li>Title 5</li> // id = 5
</ol>

我一直试图想办法如何完成这项工作。但到目前为止,每一次尝试都失败了......

任何人都知道如何从这样的数组中创建这样的嵌套<ol>列表?

请注意,我对给定数据没有任何控制权。我只是调用一个API,它返回json数据,我将其转换为数组。阵列看起来与我描述的完全一样。

3 个答案:

答案 0 :(得分:13)

您应该使用递归

首先是'php'语法中的数组:

<?php
$a=array (
    '0' => array (
        'id' => 1,
        'title' => "Title 1",
        'parent_id' => 'NULL',
        'depth' => 0
    ),
    '1' => array (
        'id' => 2,
        'title' => "Title 2",
        'parent_id' => 'NULL',
        'depth' => 0
    ),
    '2' => array (
        'id' => 3,
        'title' => "Title 3",
        'parent_id' => 2,
        'depth' => 1
    ),
    '3' => array (
        'id' => 4,
        'title' => "Title 4",
        'parent_id' => 2,
        'depth' => 1
    ),
    '4' => array (
        'id' => 5,
        'title' => "Title 5",
        'parent_id' => 'NULL',
        'depth' => 0
    ),
    '5' => array (
        'id' => 6,
        'title' => "Title 6",
        'parent_id' => 4,
        'depth' => 0
    )
);

此处代码

$level = 'NULL';

function r( $a, $level) {
   $r = "<ol>";
   foreach ( $a as $i ) {
       if ($i['parent_id'] == $level ) {
          $r = $r . "<li>" . $i['title'] . r( $a, $i['id'] ) . "</li>";
       }
   }
   $r = $r . "</ol>";
   return $r;
}

print r( $a, $level );

?>

结果

<ol><li>Title 1<ol></ol></li><li>Title 2<ol><li>Title 3<ol>
</ol></li><li>Title 4<ol><li>Title 6<ol></ol></li></ol></li></ol></li><li>Title 5
<ol></ol></li></ol>
  1. Title 1 \ n
  2. Title 2 \ n
    1. Title 3 \ n
    2. < li> Title 4 \ n
      1. Title 6 \ n
  3. Title 5 \ n

在检查为解决方案后被编辑

避免空叶:

function r( $a, $level) {
   $r = '' ;
   foreach ( $a as $i ) {
       if ($i['parent_id'] == $level ) {
          $r = $r . "<li>" . $i['title'] . r( $a, $i['id'] ) . "</li>";
       }
   }
   return ($r==''?'':"<ol>". $r . "</ol>");
}

答案 1 :(得分:4)

您可以尝试以下

$array = array (
    "0" => array (
        "id" => 1,
        "title" => "Title 1",
        "parent_id" => NULL,
        "depth" => 0
    ),
    "1" => array (
        "id" => 2,
        "title" => "Title 2",
        "parent_id" => NULL,
        "depth" => 0
    ),
    "2" => array (
        "id" => 3,
        "title" => "Title 3",
        "parent_id" => 2,
        "depth" => 1
    ),
    "3" => array (
        "id" => 4,
        "title" => "Title 4",
        "parent_id" => 2,
        "depth" => 1
    ),
    "4" => array (
        "id" => 5,
        "title" => "Title 5",
        "parent_id" => NULL,
        "depth" => 0
    ),
    "5" => array (
        "id" => 6,
        "title" => "Title 6",
        "parent_id" => 4,
        "depth" => 0
    )
);

echo(make($array));

输出

<ol>
    <li>Title 1</li>
    <li>Title 2</li>
    <ol>
        <li>Title 3</li>
        <li>Title 4</li>
        <ol>
            <li>Title 6</li>
        </ol>
    </ol>
    <li>Title 5</li>
</ol>

使用的功能

function make(array $array, $no = 0) {
    $child = hasChildren($array, $no);
    if (empty($child))
        return "";
    $content = "<ol>\n";
    foreach ( $child as $value ) {
        $content .= sprintf("\t<li>%s</li>\n", $value['title']);
        $content .= make($array, $value['id']);
    }
    $content .= "</ol>\n";
    return $content;
}

function hasChildren($array, $id) {
    return array_filter($array, function ($var) use($id) {
        return $var['parent_id'] == $id;
    });
}

See Live Demo

答案 2 :(得分:1)

以下数组:

Array
(
    [0] => Content
    [1] => Array
        (
            [0] => International
            [1] => Array
                (
                    [0] => Mexico
                    [1] => Array
                        (
                            [0] => Tamaulipas
                        )

                    [2] => USA
                )

        )

)

使用此功能:

function r($element) {
    foreach ($element as $value) {
        if (!is_array($value)) {
            echo "<li>";
            echo $value;
        } else {
            echo "<ul>";
            r($value);
            echo "</li>";
            echo "</ul>";
        }
    }
}

PHP代码:

echo "<ul>";
r($array);
echo "</ul>";

返回:

<ul>
    <li>Public</li>
    <li>User</li>
    <li>Content
        <ul>
            <li>International
                <ul>
                    <li>Mexico
                        <ul>
                            <li>Tamaulipas</li>
                        </ul>  
                    </li>
                    <li>USA</li>
                </ul>
            </li>
        </ul>
    </li>
</ul>