ThinkPHP中:使用递归写node_merge()函数
需求描述:
现有一个节点集合

可以视为一个二维数组
array(5) {
[0] => array(4) {
["id"] => string(1) "1"
["name"] => string(5) "Admin"
["title"] => string(12) "后台应用"
["pid"] => string(1) "0"
}
[1] => array(4) {
["id"] => string(1) "2"
["name"] => string(5) "Index"
["title"] => string(12) "前端应用"
["pid"] => string(1) "0"
}
[2] => array(4) {
["id"] => string(1) "3"
["name"] => string(5) "Index"
["title"] => string(12) "后台首页"
["pid"] => string(1) "1"
}
[3] => array(4) {
["id"] => string(1) "4"
["name"] => string(10) "MsgManager"
["title"] => string(12) "帖子管理"
["pid"] => string(1) "1"
}
[4] => array(4) {
["id"] => string(1) "5"
["name"] => string(4) "Rbac"
["title"] => string(10) "RBAC权限"
["pid"] => string(1) "1"
}
}
数组arr01
pid、level字段解释: pid表示父级的id。取值范围{0,1,2,3,... ,n,...}
pid的值为0,表示这条记录不属于任何其他记录。 pid的值为1,表示这条记录属于id为1的记录。即为id为1的记录的子记录。 pid的值为2,表示这条记录属于id为2的记录。即为id为2的记录的子记录。 ...... 以此类推。 level表示层次、级别。取值范围{1,2,3},即为3个层次、或者是3个级别 level的值为1,表示第一层次,最高层次级,最高级别。 level的值为2,表示第二层次。
level的值为3,表示第三层次。 例如,如果level=1代表导航栏中的首页、公司部门、产品介绍等等
那么,level=2就代表部门A、部门B、...;产品A、产品B、产品C、... 这样,仅仅靠一个二维表,就能保存下各条记录的归属关系。 但是,为了使用foreach嵌套的方式循环输出有层次的数据,我们需要把原始的二维数组,按照归属关系组合到一起。 形如:一个多维数组
array(2) {
[0] => array(5) {
["id"] => string(1) "1"
["name"] => string(5) "Admin"
["title"] => string(12) "后台应用"
["pid"] => string(1) "0"
["child"] => array(3) {
[0] => array(5) {
["id"] => string(1) "3"
["name"] => string(5) "Index"
["title"] => string(12) "后台首页"
["pid"] => string(1) "1"
["child"] => array(0) {
}
}
[1] => array(5) {
["id"] => string(1) "4"
["name"] => string(10) "MsgManager"
["title"] => string(12) "帖子管理"
["pid"] => string(1) "1"
["child"] => array(0) {
}
}
[2] => array(5) {
["id"] => string(1) "5"
["name"] => string(4) "Rbac"
["title"] => string(10) "RBAC权限"
["pid"] => string(1) "1"
["child"] => array(0) {
}
}
}
}
[1] => array(5) {
["id"] => string(1) "2"
["name"] => string(5) "Index"
["title"] => string(12) "前端应用"
["pid"] => string(1) "0"
["child"] => array(0) {
}
}
}
数组arr02
我们写一个递归函数node_merge()来实现:
<?php
/**
* 递归重组节点信息多维数组
* @param [array] $node [要处理的节点数组:二维数组]
* @param [int] $pid [父级ID]
* @return [array] [树状结构的节点体系:多维数组]
*/
function node_merge($node,$pid=0){
$arr=array(); foreach ($node as $v) {
if ($v['pid']==$pid) {
$v['child']=node_merge($node,$v['id']);
$arr[]=$v;
}
} return $arr;
}
?>
怎么理解这个函数呢? 我们以处理原始的二维数组(arr01)的数据为例,通过调用node_merge()函数,转换成可直接看出归属关系的多维数组(arr02)。 函数中的$node即为arr01,当传入arr01数组时,$pid初始化为0值,函数新建一个临时数组(arr); 接着,用foreach循环,读取第一条记录(id=1那条记录),判断式处理:如果第一条记录中的父级id是0,那么正好等于初始化的$pid, 那么,在第一条记录中,加上一个字段(child)。第一条记录中的child字段的取值,来自pid=1的记录(想想为什么。提示:从node_merge($node,$v['id']);中的$v['id']入手)。 而取值操作是来自$arr[]=$v;这句命令。 这使用的是递归,函数调用函数,这是我的理解,如果你有更好的理解,可以给我留言。邮箱:1465567571@qq.com
ThinkPHP中:使用递归写node_merge()函数的更多相关文章
- Thinkphp中controller控制器根据curl函数请求数据
public function member(){ $url="http://aitequn.tjnit.com/UserAction_findAllUser"; $ch =cur ...
- C#中的函数式编程:递归与纯函数(二)
在序言中,我们提到函数式编程的两大特征:无副作用.函数是第一公民.现在,我们先来深入第一个特征:无副作用. 无副作用是通过引用透明(Referential transparency)来定义的.如果一个 ...
- Thinkphp中的 I 函数(Thinkphp3.2.3版本)
I 函数的作用是获取系统变量,必要时还可以对变量值进行过滤及强制转化,I 函数的语法格式: I('变量类型.变量名/修饰符',['默认值'],['过滤方法或正则'],['额外数据源']) 一.获取变量 ...
- C#中的函数式编程:递归与纯函数(二) 学习ASP.NET Core Razor 编程系列四——Asp.Net Core Razor列表模板页面
C#中的函数式编程:递归与纯函数(二) 在序言中,我们提到函数式编程的两大特征:无副作用.函数是第一公民.现在,我们先来深入第一个特征:无副作用. 无副作用是通过引用透明(Referential ...
- 有一字符串,包含n个字符。写一函数,将此字符串中从第m个字符开始的全部字符复制成为另一个字符串。
[提交][状态][讨论版] 题目描述 有一字符串,包含n个字符.写一函数,将此字符串中从第m个字符开始的全部字符复制成为另一个字符串. 输入 数字n 一行字符串 数字m 输出 从m开始的子串 样例输入 ...
- 面试题-->写一个函数,返回一个数组中所有元素被第一个元素除的结果
package com.rui.test; import java.util.Random; /** * @author poseidon * @version 1.0 * @date:2015年10 ...
- HTML中直接写js 函数
1.在HTML中直接写JS函数: <body onload="javascript:{window.location.href='http://www.baidu.com/'}&quo ...
- C语言:写一函数,将两个字符串中的元音字母复制到另一个字符串,然后输出
题目描述 写一函数,将两个字符串中的元音字母复制到另一个字符串,然后输出. 输入 一行字符串 输出 顺序输出其中的元音字母(aeiuo) 样例输入 abcde 样例输出 ae 编码: #include ...
- 38 写一个函数,求一个字符串的长度,在main函数中输入字符串,并输出其长度。
题目:写一个函数,求一个字符串的长度,在main函数中输入字符串,并输出其长度. public class _038PrintLength { public static void main(Stri ...
随机推荐
- 关于JDBC导入mysql的jar驱动的头痛
今天上午想写个小程序,需要调用数据库,查了书和各个博客. 最后卡在导入mysql驱动上了,花了1个多小时才让程序连上数据库. 这里有个小误区,你下载的是zip压缩文件,很多帖子写的都是让你导入驱动,但 ...
- CRF技能词识别过程
最近在用CRF做未登录技能词识别,虽然艰难,但是感觉很爽,效率非常高. (1)数据准备: 选取30000行精语料作为训练数据.每一个br作为一条数据.使用已有的技能词典对数据进行无标注分词. (2)训 ...
- 挖个坑,写一个Spring+SpringMVC+Mybatis的项目
想挖个坑督促自己练技术,有时候想到一个项目,大概想了一些要实现的功能,怎么实现.现在觉得自己差不多能完成QQ空间的主要功能了.准备立个牌坊,写一个类似功能的网站.并且把进度放到这里来. 初步计划实现以 ...
- 早期MyBatis开发与接口式Mybatis开发的简介
早期MyBatis开发与接口式Mybatis开发的简介 一.早期版本的myBatis使用 导jar包 1.配置mybatis.xml的配置文件 1) ...
- NCS8801S芯片RGB/LVDS转EDP功能简介
NCS8801S RGB/LVDS-to-eDP Converter (1/2/4-lane eDP) Features --Embedded-DisplayPort (eDP) Output 1/2 ...
- SSH2项目网上书店系统手把手教学_Struts2+Spring+Hibernate整合开发
一 序言 鉴于目前J2EE的火热程度,SSH2是每个学生毕业前都必须掌握的一门技术,所以在这里我就使用SSH2技术做一个小型项目,和大家一起学习. SSH2技术的基础概论就不再提了,直接说特点吧. 1 ...
- jsp、freemarker、velocity区别详解
详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt172 在java领域,表现层技术主要有三种:jsp.freemarker.v ...
- 使用BootStrap框架设置全局CSS样式
一.排版 标题 HTML 中的所有标题标签,<h1> 到 <h6> 均可使用.另外,还提供了 .h1 到 .h6 类,为的是给内联(inline)属性的文本赋予标题的样式. & ...
- Linux开机最简化
[root@localhost ~]# LANG=en [root@localhost ~]# for root in chkconfig --list|grep 3:on|awk '{print $ ...
- 团队作业八——第二次团队冲刺(Beta版本)第4天
团队作业八--第二次团队冲刺(Beta版本)第4天 一.每个人的工作 (1) 昨天已完成的工作 做一下用户注册的功能和登录功能. (2) 今天计划完成的工作 完成界面跳转 (3) 工作中遇到的困难 界 ...