当前位置:编程学习 > php >>

无限递归树展示

[php] 
<?php 
/**
 * 无限级(受尾节点描述算法限制, 详见tree_parse注释)递归菜单
 * author: selfimpr
 * blog: http://blog.csdn.net/lgg201
 * mail: lgg860911@yahoo.com.cn
 */ 
 
define('MAX_NODES',         3);                     /* 最大子节点数 */ 
define('MAX_NODE_INDEX',    MAX_NODES - 1);         /* 子节点最大索引值 */ 
define('NAME_FMT',          'name-%08d');               /* 节点内容输出格式串 */ 
 
/* 树节点数据结构 */ 
define('K_ID',              'id'); 
define('K_NAME',            'name'); 
define('K_CHILD',           'children'); 
 
/* 输出构造时使用的拼装字符 */ 
define('PREFIX_TOP',        '┏');                   /* 第一层第一个节点的标识符 */ 
define('PREFIX_BOTTOM',     '┗');                   /* 每一个父节点的最后一个子节点的标识符 */ 
define('PREFIX_MIDDLE',     '┠');                   /* 所有非上面两种情况的节点的标识符 */ 
define('PREFIX_LINE',       '┇');                   /* 祖先节点的连线符 */ 
define('SPACE',             ' ');                   /* 空白占位(所有尾节点不显示连线符) */ 
define('WIDE_SPACE',        str_repeat(SPACE, 4));  /* 宽的空白占位, 为了让树的层次清晰 */ 
 
 
/**
 * data_build 
 * 构造一个节点
 * @param mixed $id         节点id
 * @param mixed $is_leaf    是否叶子
 * @access public
 * @return void
 */ 
function node_build($id, $is_leaf = FALSE) { 
    return array( 
        K_ID    => $id,  
        K_NAME  => sprintf(NAME_FMT, $id),  
        K_CHILD => $is_leaf ? NULL : array(),  
    ); 

/**
 * tree_build 
 * 构造一棵树(树中每个节点的子节点数由MAX_NODES确定)
 * @param mixed $datas  要返回的树引用
 * @param mixed $id     起始ID
 * @param mixed $level  树的层级
 * @access public
 * @return void
 */ 
function tree_build(&$datas, &$id, $level) { 
    if ( $level < 1 ) return ; 
    $is_leaf    = $level == 1; 
    $i          = -1; 
    $next_level = $level - 1; 
    while ( ++ $i < MAX_NODES ) { 
        $data   = node_build($id ++, $is_leaf); 
        if ( !$is_leaf )  
            tree_build($data[K_CHILD], $id, $next_level); 
        array_push($datas, $data); 
    } 

 
/**
 * node_str 
 * 输出一个节点自身的信息
 * @param mixed $string 返回结果的字符串(引用传值)
 * @param mixed $data   节点数据
 * @access public
 * @return void
 */ 
function node_str(&$string, $data) { 
    $string .= sprintf(' %s[%d]', $data[K_NAME], $data[K_ID]); 

/**
 * node_sign 
 * 输出一个节点的标志符号
 * @param mixed $string 返回结果的字符串(引用传值)
 * @param mixed $level  当前深度
 * @param mixed $i      当前节点在父节点中的索引(下标)
 * @access public
 * @return void
 */ 
function node_sign(&$string, $level, $i) { 
    switch ( $i ) { 
        case 0: 
            $string .= $level == 0 ? PREFIX_TOP : PREFIX_MIDDLE; 
            break; 
        case MAX_NODE_INDEX: 
            $string .= PREFIX_BOTTOM; 
            break; 
        default: 
            $string .= PREFIX_MIDDLE; 
            break; 
    } 

/**
 * node_prefix 
 * 输出一个节点的前缀
 * @param mixed $string     返回结果的字符串(引用传值)
 * @param mixed $level      当前深度
 * @param mixed $is_last    当前节点(含)所有祖先节点是否尾节点标记
 * @access public
 * @return void
 */ 
function node_prefix(&$string, $level, $is_last) { 
    if ( $level > 0 ) { 
        $i  = 0; 
        /* 前缀格式: "父级连线" ["宽空白符" "父级连线" ...] "宽空白符" */ 
        $string .= ($is_last & 1 << ($level - $i) ? SPACE : PREFIX_LINE); 
        while ( ++ $i < $level )  
            $string .= WIDE_SPACE . ($is_last & 1 << ($level - $i) ? SPACE : PREFIX_LINE); 
        $string .= WIDE_SPACE; 
    } 

/**
 * node_out 
 * 输出一个节点
 * @param mixed $string     返回结果的字符串(引用传值)
 * @param mixed $data       要处理的节点数据
 * @param mixed $level      节点深度<
补充:Web开发 , php ,
CopyRight © 2022 站长资源库 编程知识问答 zzzyk.com All Rights Reserved
部分文章来自网络,