MySQL树形结构的数据库表设计和查询
1、邻接表(Adjacency List)
- 实例:现在有一个要存储一下公司的人员结构,大致层次结构如下:
那么怎么存储这个结构?并且要获取以下信息:
- 1.查询小天的直接上司。
- 2.查询老宋管理下的直属员工。
- 3.查询小天的所有上司。
- 4.查询老王管理的所有员工。
方案一、(Adjacency List)只存储当前节点的父节点信息。
-- -- MySQL树结构 --
-- Author: xielong Email:cnxielong@gmail.com -- -- 建表 --
DROP TABLE IF EXISTS Employees; CREATE TABLE IF NOT EXISTS Employees (
id INT AUTO_INCREMENT,
ename VARCHAR (),
job VARCHAR (),
parent_id INT,
PRIMARY KEY(id)
) ENGINE = INNODB DEFAULT CHARSET = UTF8; DESCRIBE Employees -- 插入数据 --
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('老王','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('老宋','产品部主管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('老牛','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('小吴','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('小李','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('小欢','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('小小','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('小天','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('肖丽','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('十号','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('十一号','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('十二号','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('十三号','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('十四号','高管','');
INSERT INTO `employees` ( `ename`, `job`, `parent_id`) VALUES('小黑十五','高管','');
数据库结构信息:
好的,现在开始进入回答环节:
- 1.查询小天的直接上司:
-- 查询小天的上级Id 隐式内连接 关键条件父节点 --
SELECT e2.id, e2.ename, e2.job FROM Employees e1, Employees e2 WHERE e1.`parent_id` = e2.id AND e1.id= -- 查询小天的上级Id 显示内连接 关键条件父节点 --
SELECT e2.id, e2.ename, e2.job FROM Employees e1 INNER JOIN Employees e2 WHERE e1.`parent_id` = e2.id AND e1.id=
2.查询老宋管理下的直属员工:
-- 查询老宋管理下的直属员工:隐式内连接 关键条件父子节点 老宋ID= --
SELECT e2.id, e2.ename, e2.job FROM Employees e1, Employees e2 WHERE e1.id= AND e2.`parent_id` = e1.id -- 查询小天的上级Id 显示内连接 关键条件父子节点 老宋.ename='老宋' --
SELECT e2.id, e2.ename, e2.job FROM Employees e1 INNER JOIN Employees e2 WHERE e2.`parent_id` = e1.id AND e1.ename='老宋'
3.查询小天的所有上司。
- 这里肯定没法直接查,只能用循环进行循环查询,先查直接上司,再查直接上司的直接上司,依次循环,这样麻烦的事情,还是得先建立一个存储过程:
- 睁大眼睛看仔细了,接下来是骚操作环节:
-- 、查询小天的所有上级 -- -- 删除 --
DROP FUNCTION IF EXISTS getSuperiors; -- 创建 --
DELIMITER $$ CREATE DEFINER=`root`@`localhost` FUNCTION `getSuperiors` (`uid` INT) RETURNS VARCHAR()
BEGIN
DECLARE superiors VARCHAR() DEFAULT '';
DECLARE sTemp INTEGER DEFAULT uid;
DECLARE tmpName VARCHAR(); WHILE (sTemp>) DO
SELECT parent_id INTO sTemp FROM employees WHERE id = sTemp;
SELECT ename INTO tmpName FROM employees WHERE id = sTemp;
IF(sTemp>)THEN
SET superiors = CONCAT(tmpName,',',superiors);
END IF;
END WHILE;
SET superiors = LEFT(superiors,CHARACTER_LENGTH(superiors)-);
RETURN superiors;
END $$ -- 调用 --
SELECT getSuperiors() 上司;
这一段存储过程可以查询子节点的所有父节点,来试验一下
好的,骚操作完成。
显然,这样。获取子节点的全部父节点的时候很麻烦。。
4.查询老王管理的所有员工。
思路如下:先获取所有父节点为老王id的员工id,然后将员工姓名加入结果列表里,在调用一个神奇的查找函数,即可进行神奇的查找:
-- 查询老王管理的所有员工 -- -- 删除 --
DROP PROCEDURE IF EXISTS getSubordinate -- 创建 --
DELIMITER $$ CREATE DEFINER = `root` @`localhost` FUNCTION `getSubordinate` (`uid` INT) RETURNS VARCHAR ()
BEGIN
DECLARE str VARCHAR ();
DECLARE cid VARCHAR ();
DECLARE result VARCHAR ();
DECLARE tmpName VARCHAR ();
SET str = '$';
SET cid = CAST(uid AS CHAR());
WHILE
cid IS NOT NULL DO
SET str = CONCAT(str, ',', cid);
SELECT GROUP_CONCAT(id) INTO cid FROM employees WHERE FIND_IN_SET(parent_id, cid);
END WHILE;
SELECT
GROUP_CONCAT(ename) INTO result FROM employees WHERE FIND_IN_SET(parent_id, str);
RETURN result;
END $$ -- 调用 --
SELECT getSubordinate()
看神奇的结果:
虽然搞出来了,但说实话,真是不容易。。。
这种方法的优点是存储的信息少,查直接上司和直接下属的时候很方便,缺点是多级查询的时候很费劲。所以当只需要用到直接上下级关系的时候,用这种方法还是不错的,可以节省很多空间。
2、继承关系驱动的设计表和基于左右值编码的设计
参考链接: https://blog.csdn.net/lj1314ailj/article/details/52074216
参考:MySQL 实现树形的遍历
MySQL 实现树形的遍历(关于多级菜单栏以及多级上下部门的查询问题)
参考链接: https://blog.csdn.net/mchdba/article/details/39277301 ---------------------
MySQL树形结构的数据库表设计和查询的更多相关文章
- 树形结构的数据库表Schema设计-基于左右值编码
树形结构的数据库表Schema设计 程序设计过程中,我们常常用树形结构来表征某些数据的关联关系,如企业上下级部门.栏目结构.商品分类等等,通常而言,这些树状结构需要借助于数据库完 成持久化.然而目前的 ...
- 树形结构的数据库表Schema设计
今天又有幸遇到一个不知道的东西,那就是树型结构在数据库表中设计的问题.由于只是阅读了人家的东西,就直接给连接吧. 第一个:http://blog.csdn.net/monkey_d_meng/arti ...
- 数据库表设计时一对一关系存在的必要性 数据库一对一、一对多、多对多设计 面试逻辑题3.31 sql server 查询某个表被哪些存储过程调用 DataTable根据字段去重 .Net Core Cors中间件解析 分析MySQL中哪些情况下数据库索引会失效
数据库表设计时一对一关系存在的必要性 2017年07月24日 10:01:07 阅读数:694 在表设计过程中,我无意中觉得一对一关系觉得好没道理,直接放到一张表中不就可以了吗?真是说,网上信息什么都 ...
- Innodb IO优化 — 数据库表设计 转
数据库表设计这块学问比较多,我这里单从互联网角度出发同时结合Innodb的特性给出一些设计方法供大家参考.本文构建大概分两分部分:Innodb的特性及设计中如何利用这种特性. Innodb特性: In ...
- MySQL 树形结构 根据指定节点 获取其所在全路径节点序列
背景说明 需求:MySQL树形结构, 根据指定的节点,获取其所在全路径节点序列. 问题分析 1.可以使用类似Java这种面向对象的语言,对节点集合进行逻辑处理,获取全路径节点序列. 2.直接自定义My ...
- MySQL 树形结构 根据指定节点 获取其所有父节点序列
背景说明 需求:MySQL树形结构, 根据指定的节点,获取其所有父节点序列. 问题分析 1.可以使用类似Java这种面向对象的语言,对节点集合进行逻辑处理,获取父节点. 2.直接自定义MySQL函数 ...
- MySQL 树形结构 根据指定节点 获取其下属的所有子节点(包含路径上的枝干节点和叶子节点)
背景说明 需求:MySQL树形结构, 根据指定的节点,获取其下属的所有子节点(包含路径上的枝干节点和叶子节点) 枝干节点:如果一个节点下还有子节点,则为枝干节点. 叶子节点:如果一个节点下不再有子节点 ...
- MySQL 树形结构 根据指定节点 获取其所有叶子节点
背景说明 需求:MySQL树形结构, 根据指定的节点,获取其下属的所有叶子节点. 叶子节点:如果一个节点下不再有子节点,则为叶子节点. 问题分析 1.可以使用类似Java这种面向对象的语言,对节点集合 ...
- Oracle数据库表设计时的注意事项
表是Oracle数据库中最基本的对象之一.万丈高楼从平地起,这个基础对象对于数据库来说,非常重要.因为其设计是否合理,直接跟数据库的性能相关.从Oracle数据库菜鸟到数据库专家这个过程中,在表设计与 ...
随机推荐
- uboot对Flash和DDR的分区管理
1.uboot阶段对Flash的分区 (1).所谓分区,就是对Flash进行分块管理. (2).PC机等产品中,因为大家都是在操作系统下使用硬盘的,整个硬盘由操作系统统一管理,操作系统会使用文件系统帮 ...
- pandas 学习笔记【持续更新】
import numpy as np import pandas as pd import matplotlib.pyplot as plt df1 = pd.DataFrame(np.arange( ...
- goweb- session和数据存储
session和数据存储 Web开发中一个很重要的议题就是如何做好用户的整个浏览过程的控制,因为HTTP协议是无状态的,所以用户的每一次请求都是无状态的,我们不知道在整个Web操作过程中哪些连接与该用 ...
- ZZJ_淘淘商城项目:day03(淘淘商城02 - 后台系统功能实现)
1. 今日大纲 1. 学习Nginx的使用 2. 实现商品的管理 a) 新增商品 b) 查询商品列表 c) 编辑商品 d) 删除商品 e) ...
- Java多线程常见概念
进程和线程的区别 进程是资源分配的最小单位,线程是CPU调度的最小单位 线程不能看做独立应用,而进程可以 进程有独立的地址空间,互相不影响,线程只是进程的不同执行路径 线程没有独立的地址空间,多进程的 ...
- 吴裕雄--天生自然C语言开发:判断
if(boolean_expression) { /* 如果布尔表达式为真将执行的语句 */ } #include <stdio.h> int main () { /* 局部变量定义 */ ...
- 一个简单的jQuery回调函数例子
jQuery回调函数简单使用 比如说,我们想要点击某个按钮后触发事件, 先把一些指定内容给隐藏掉, 然后跳出相关信息的对话框. 如果使用普通的方法, 不用回调函数的话, 会有怎么样的效果呢? 效果是先 ...
- 1059 C语言竞赛 (20 分)
题目:1059 C语言竞赛 (20 分) C 语言竞赛是浙江大学计算机学院主持的一个欢乐的竞赛.既然竞赛主旨是为了好玩,颁奖规则也就制定得很滑稽: 0.冠军将赢得一份“神秘大奖”(比如很巨大的一本学生 ...
- iOS渐变视图&动画库、腰杆、音频水滴水波手势、多种对话框、四级展开效果等源码
iOS精选源码 用户行为追踪--无侵入埋点 .终端日志的打印 支持storyboard的渐变视图&动画库 支持圆形.竖直.横向的摇杆 纯swift实现的类似excel表格效果 swift实现自 ...
- C++头文件和std命名空间
C++ 是在C语言的基础上开发的,早期的 C++ 还不完善,不支持命名空间,没有自己的编译器,而是将 C++ 代码翻译成C代码,再通过C编译器完成编译.这个时候的 C++ 仍然在使用C语言的库,std ...