PHP在无限分类时注意的一些问题(不保证代码完全正确哦)
1 无限分类的查找(获取所有节点)
代码:
/**
* 无限分类查询,默认 pid 为 0
* @param $pid
* @return array $res
*/
protected function selectTree($pid = 0)
{
$res = [];
$sql = "SELECT * FROM " . $this->tbname . " WHERE pid=" . $pid;
$result = @mysqli_query($this->link, $sql);
if ($result) {
$count = mysqli_num_rows($result);
if ($count > 0) {
while ($rows = mysqli_fetch_assoc($result)) {
$rows['children'] = $this->selectTree($rows['id']);
$res[] = $rows;
}
}
mysqli_free_result($result);
}
return $res;
}
2 无限分类节点的删除,不能单纯地删除当前节点,需要查找到当前节点下的所有子节点,一并删除
代码:
/**
* 删除目录树
* @param $id
*/
protected function deleteTree($id)
{
$res = $this->selectTree($id);
if (!empty($res)) {
foreach ($res as $v) {
$this->deleteTree($v['id']);
}
}
$sql = "DELETE FROM " . $this->tbname . " WHERE id=" . $id;
mysqli_query($this->link, $sql);
}
3 **无限分类的编辑,由于在编辑的时候其父级是可选择的,所以有可能造成用户选择到当前节点的子节点(们),所以要进行判断。虽说是无限分类,但正常情况下目录深度是会有限度的,如果给定了目录深度,还要判断选择父级之后的目录深度是否超出范围。
如果将编辑的元素放在其子元素下,所造成的问题:在查询的时候无限循环!!
|
id |
pid |
name |
|
1 |
0 |
test1 |
|
2 |
1 |
test2 |
修改之后:
|
id |
pid |
name |
|
1 |
2 |
test1 |
|
2 |
1 |
test2 |
如上表所示,在进行无限分类查询时,就会陷入死循环!
所以,针对可能会出现的问题,给出下面的解决办法,在用户修改时进行判断,通过则可以修改,未通过则给出提示。
3.1 判断用户选择的是否是当前节点(这个只需要判断选择的节点和当前编辑节点的ID是否相同即可)
3.2 判断用户选择的是否是子节点(如果是的话返回true)
/**
* 判断id所对应的元素是否是pid所对应元素的子元素,是的话返回true
* @param $id
* @param $pid
* @return boolean $result
*/
protected function isChild($id, $pid)
{
$result = false;
$sql = "SELECT pid FROM " . $this->tbname . " WHERE id=" . $id;
$res = @mysqli_query($this->link, $sql);
if ($res) {
while ($rows = mysqli_fetch_assoc($res)) {
$result = ($pid === $rows['pid']) ? true : (($rows['pid'] !== 0) ? $this->isChild($rows['pid'], $pid) : false);
}
}
return $result;
}
3.3 判断用户选择的节点是否已经达到目录深度
在做完后面的一步之后,这一步就比较好实现了:
/**
* 判断所选元素是否达到目录深度,达到返回true
* @param $id
* @return mixed
*/
protected function isMaxDeep($id)
{
return $this->deepUp($id) >= $this->maxDeep;
}
3.4 判断修改之后的目录深度是否超出限定范围
/**
* 修改之后的最终深度,如果深度大于规定深度,返回true
* @param $pid
* @param $id
* @return mixed
*/
protected function lastDeep($pid, $id)
{
return ($this->deepUp($pid) + $this->deepDown($id)) > $this->maxDeep;
} /**
* 向上查找父元素的深度
* @param $id
* @param int $k
* @return int
*/
protected function deepUp($id, $k = 1)
{
$sql = "SELECT pid FROM " . $this->tbname . "WHERE id=" . $id;
$res = @mysqli_query($this->link, $sql);
if ($res) {
while ($rows = mysqli_fetch_assoc($res)) {
($rows['pid'] !== 0) && $k = $this->deepUp($rows['pid'], $k+1);
}
}
return $k;
} /**
* 向下查找子元素的深度
* @param $id
* @param int $k
* @return int
*/
protected function deepDown($id, $k = 0)
{
$sql = "SELECT * FROM " . $this->tbname . "WHERE pid=" . $id;
$res = @mysqli_query($this->link, $sql);
if ($res && mysqli_num_rows($res) > 0) {
$k++;
while ($rows = mysqli_fetch_assoc($res)) {
$k = max($k, $this->deepDown($rows['id'], $k))
}
}
return $k;
}
经过上面的判断之后,根据返回的结果就能判断是否可以修改,如果返回true,则不可以修改,如果是false则可以进行修改。
(如果不用无限分类查询,只是普通的查询,让前端去实现结果的显示会怎么样呢??不懂那些框架是怎么实现的,感觉也是在用递归)
PHP在无限分类时注意的一些问题(不保证代码完全正确哦)的更多相关文章
- MySql无限分类数据结构--预排序遍历树算法
MySql无限分类数据结构--预排序遍历树算法 无限分类是我们开发中非常常见的应用,像论坛的的版块,CMS的类别,应用的地方特别多. 我们最常见最简单的方法就是在MySql里ID ,parentID, ...
- TreeView递归绑定无限分类数据
TreeView递归绑定无限分类数据 实现一个动态绑定,无限级分类数据时,需要将数据绑定到TreeView控件,分类表的结构是这样的: 字段 类型 Id int ParentId int Name N ...
- PHP全路径无限分类原理
全路径无限分类:以一个字段把他所有的父级id按顺序记录下来以此实现的无限分类叫做全路径无限分类 优点:查询方便 缺点:增加,移动分类时数据维护时稍微复杂.
- PHP实现无限分类
PHP实现无限分类 无限分类 递归 无限级分类是一种设计技巧,在开发中经常使用,例如:网站目录.部门结构.文章分类.笔者觉得它在对于设计表的层级结构上面发挥很大的作用,比如大家在一些平台上面,填写邀请 ...
- laravel-nestedset:多级无限分类正确姿势
laravel-nestedset:多级无限分类正确姿势 laravel-nestedset是一个关系型数据库遍历树的larvel4-5的插件包 目录: Nested Sets Model简介 安 ...
- 后台树状菜单,js实现递归无限分类
//新闻类别管理 public function new_classify() { $arr = M('news_classify')->where("fid = 0")-& ...
- 关于无限分类的树状输出(id,name,pid)类型的
首先创建无限分类的数据表,我这里采用的是id.name.pid这种类型(当然还有很多种无限分类的方式了,比如:id.name.pid.path.left.right左右节点的形式) CREATE TA ...
- php递归方法实现无限分类实例
数组: 代码如下 复制代码 $items = array( array('id' => 1, 'pid' => 0, 'name' => '一级11' ), array('id' ...
- PHP+Mysql无限分类的方法汇总
无限分类是个老话题了,来看看PHP结合Mysql如何实现.第一种方法这种方法是很常见.很传统的一种,先看表结构表:categoryid int 主键,自增name varchar 分类名称pid in ...
随机推荐
- Vue模板语法(一)
Vue模板语法 一.插值 1.1.1 文本 {{msg}} 1.1.2 html 使用v-html指令用于输出html代码 1.1.3 属性 HTML属性中的值应使用v-bind指令 1.1.4 表达 ...
- selenium--多窗口操作
前戏 想一想,我们为什么要获取窗口句柄呢?有什么用呢? 来假设一下,我们打开了一个网站,点击了一个按钮,新打开了一个页面,我们在新页面操作完成之后,需要回到原来的页面继续操作,这时候你如果继续操作原来 ...
- luoguP2173 [ZJOI2012]网络 LCT
链接 luogu 思路 颜色很少,开10个lct分别维护 if (Hash.count(make_pair(u, v)) && Hash[make_pair(u, v)] == col ...
- 基于Linux(中标麒麟)上QT的环境搭建
最近由于公司需要,需要在中标麒麟上进行QT的二次开发,但是网上的资料很少,就算是有也基本都是其他的版本的Linux上的搭建.中标麒麟本身的资料也很好,而且还只能试用60天. 下面就介绍下我对此环境的搭 ...
- pytest 打印调试信息
pytest_lean2.py #coding=utf- import pytest import os import sys import time import json sys.path.app ...
- Redis读写分离技术解析
背景 云数据库Redis版不管主从版还是集群规格,replica作为备库不对外提供服务,只有在发生HA的时候,replica提升为master后才承担读写流量.这种架构读写请求都在master上完成, ...
- Linux文件和目录管理
一.与文档相关的命令 1,命令head:用于显示文件前10行,后面直接跟文件名.如果加-n,则显示文件的前几行. 选项-n后有无空格均可也可以省略字母n,直接跟数字 2,命令tail:和命令head类 ...
- samba服务器安装测试
samba服务器安装 1.简介 作用:samba文件服务器允许通过网络跨步同的操作系统进行文件共享. 2.安装samba服务器 $ sudo apt-get update $ sudo apt-get ...
- java ---- 认识类和对象
---- 开发方法 结构化开发方法特点: 1.面向功能划分软件结构 2.自顶而下 3.最小的子系统是方法 4.制约了软件的可维护性和扩展性 面向对象开发方法特点: 1.把软件系统看成各种对象和集合 ...
- 低版本的 eclipse 不支持 tomcat8.5 的解决方法
目录 低版本的eclipse 不支持 tomcat8.5,但是还想使用的方法 低版本的eclipse 不支持 tomcat8.5,但是还想使用的方法 1. 介绍: 我在使用 mars 版本的 ec ...