欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談MYSQL中樹形結(jié)構(gòu)表3種設(shè)計(jì)優(yōu)劣分析與分享

 更新時(shí)間:2021年09月22日 10:32:33   作者:程序員小強(qiáng)  
在開發(fā)中經(jīng)常遇到樹形結(jié)構(gòu)的場景,本文將以部門表為例對比幾種設(shè)計(jì)的優(yōu)缺點(diǎn),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下

簡介

在開發(fā)中經(jīng)常遇到樹形結(jié)構(gòu)的場景,本文將以部門表為例對比幾種設(shè)計(jì)的優(yōu)缺點(diǎn);

問題

需求背景:根據(jù)部門檢索人員,
問題:選擇一個頂級部門情況下,跨級展示當(dāng)前部門以及子部門下的所有人員,表怎么設(shè)計(jì)更合理 ?

image.png

遞歸嗎 ?遞歸可以解決,但是勢必消耗性能

設(shè)計(jì)1:鄰接表

注:(常見父Id設(shè)計(jì))

表設(shè)計(jì)

CREATE TABLE `dept_info01` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT COMMENT '自增主鍵',
  `dept_id` int(10) NOT NULL COMMENT '部門id',
  `dept_name` varchar(100) NOT NULL COMMENT '部門名稱',
  `dept_parent_id` int(11) NOT NULL COMMENT '父部門id',
  `create_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '創(chuàng)建時(shí)間',
  `update_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '修改時(shí)間',
  PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

image.png

這樣是最常見的設(shè)計(jì),能正確的表達(dá)菜單的樹狀結(jié)構(gòu)且沒有冗余數(shù)據(jù),但在跨層級查詢需要遞歸處理。

SQL示例

1.查詢某一個節(jié)點(diǎn)的直接子集

SELECT * FROM dept_info01  WHERE dept_parent_id =1001

優(yōu)點(diǎn)

結(jié)構(gòu)簡單 ;

缺點(diǎn)

1.不使用遞歸情況下無法查詢某節(jié)點(diǎn)所有父級,所有子集

設(shè)計(jì)2:路徑枚舉

在設(shè)計(jì)1基礎(chǔ)上新增一個父部門id集字段,用來存儲所有父集,多個以固定分隔符分隔,比如逗號。

表設(shè)計(jì)

CREATE TABLE `dept_info02` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT COMMENT '自增主鍵',
  `dept_id` int(10) NOT NULL COMMENT '部門id',
  `dept_name` varchar(100) NOT NULL COMMENT '部門名稱',
  `dept_parent_id` int(11) NOT NULL COMMENT '父部門id',
  `dept_parent_ids` varchar(255) NOT NULL DEFAULT '' COMMENT '父部門id集',
  `create_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '創(chuàng)建時(shí)間',
  `update_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '修改時(shí)間',
  PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

image.png

SQL示例

1.查詢所有子集
1).通過模糊查詢

SELECT
 *
FROM
	dept_info02
WHERE
	dept_parent_ids like '%1001%'

2).推薦使用 FIND_IN_SET 函數(shù)

SELECT
	* 
FROM
	dept_info02 
WHERE
	FIND_IN_SET( '1001', dept_parent_ids )

優(yōu)點(diǎn)

  • 方便查詢所有的子集 ;
  • 可以因此通過比較字符串dept_parent_ids長度獲取當(dāng)前節(jié)點(diǎn)層級 ;

缺點(diǎn)

  • 新增節(jié)點(diǎn)時(shí)需要將dept_parent_ids字段值處理好 ;
  • dept_parent_ids字段的長度很難確定,無論長度設(shè)為多大,都存在不能夠無限擴(kuò)展的情況 ;節(jié)
  • 點(diǎn)移動復(fù)雜,需要同時(shí)變更所有子集中的dept_parent_ids字段值 ;

設(shè)計(jì)3:閉包表

  • 閉包表是解決分級存儲的一個簡單而優(yōu)雅的解決方案,這是一種通過空間換取時(shí)間的方式 ;
  • 需要額外創(chuàng)建了一張TreePaths表它記錄了樹中所有節(jié)點(diǎn)間的關(guān)系 ;
  • 包含兩列,祖先列與后代列,即使這兩個節(jié)點(diǎn)之間不是直接的父子關(guān)系;同時(shí)增加一行指向節(jié)點(diǎn)自己 ;

表設(shè)計(jì)

主表

CREATE TABLE `dept_info03` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT COMMENT '自增主鍵',
  `dept_id` int(10) NOT NULL COMMENT '部門id',
  `dept_name` varchar(100) NOT NULL COMMENT '部門名稱',
  `create_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '創(chuàng)建時(shí)間',
  `update_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '修改時(shí)間',
  PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

image.png

祖先后代關(guān)系表

CREATE TABLE `dept_tree_path_info` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT COMMENT '自增主鍵',
  `ancestor` int(10) NOT NULL COMMENT '祖先id',
  `descendant` int(10) NOT NULL COMMENT '后代id',
  `depth` tinyint(4) NOT NULL DEFAULT '0' COMMENT '層級深度',
  PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

注:depth 層級深度字段 ,自我引用為 1,直接子節(jié)點(diǎn)為 2,再一下層為 3,一次類推,第幾層就是幾 。

image.png

SQL示例

插入新節(jié)點(diǎn)

INSERT INTO dept_tree_path_info (ancestor, descendant,depth)
SELECT t.ancestor, 3001,t.depth+1 FROM dept_tree_path_info AS t 
WHERE t.descendant = 2001
UNION ALL
SELECT 3001,3001,1

查詢所有祖先

SELECT
	c.*
FROM
	dept_info03 AS c
INNER JOIN dept_tree_path_info t ON c.dept_id = t.ancestor
WHERE
	t.descendant = 3001

查詢所有后代

SELECT
	c.*
FROM
	dept_info03 AS c
INNER JOIN dept_tree_path_info t ON c.dept_id = t.descendant
WHERE
t.ancestor = 1001

刪除所有子樹

DELETE 
FROM
	dept_tree_path_info 
WHERE
	descendant IN 
	( 
		SELECT
			a.dept_id 
		FROM
		( SELECT descendant dept_id FROM dept_tree_path_info WHERE  ancestor = 1001 ) a
	)

刪除葉子節(jié)點(diǎn)

DELETE 
FROM
	dept_tree_path_info 
WHERE
	descendant = 2001

移動節(jié)點(diǎn)

  • 刪除所有子樹(先斷開與原祖先的關(guān)系)
  • 建立新的關(guān)系

優(yōu)點(diǎn)

  • 非遞歸查詢減少冗余的計(jì)算時(shí)間 ;
  • 方便非遞歸查詢?nèi)我夤?jié)點(diǎn)所有的父集 ;
  • 方便查詢?nèi)我夤?jié)點(diǎn)所有的子集 ;
  • 可以實(shí)現(xiàn)無限層級 ;
  • 支持移動節(jié)點(diǎn) ; 

缺點(diǎn)

  • 層級太多情況下移動樹節(jié)點(diǎn)會帶來關(guān)系表多條操作 ;
  • 需要單獨(dú)一張表存儲對應(yīng)關(guān)系,在新增與編輯節(jié)點(diǎn)時(shí)操作相對復(fù)雜 ;

結(jié)合使用

可以將鄰接表方式與閉包表方式相結(jié)合使用。實(shí)際上就是將父id冗余到主表中,在一些只需要查詢直接關(guān)系的業(yè)務(wù)中就可以直接查詢主表,而不需要關(guān)聯(lián)2張表了。在需要跨級查詢時(shí)祖先后代關(guān)系表就顯得尤為重要。

表設(shè)計(jì)

主表

CREATE TABLE `dept_info04` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT COMMENT '自增主鍵',
  `dept_id` int(10) NOT NULL COMMENT '部門id',
  `dept_name` varchar(100) NOT NULL COMMENT '部門名稱',
  `dept_parent_id` int(11) NOT NULL COMMENT '父部門id',
  `create_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '創(chuàng)建時(shí)間',
  `update_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '修改時(shí)間',
  PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

祖先后代關(guān)系表

CREATE TABLE `dept_tree_path_info` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT COMMENT '自增主鍵',
  `ancestor` int(10) NOT NULL COMMENT '祖先id',
  `descendant` int(10) NOT NULL COMMENT '后代id',
  `depth` tinyint(4) NOT NULL DEFAULT '0' COMMENT '層級深度',
  PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

總結(jié)

其實(shí),在以往的工作中,曾見過不同類型的設(shè)計(jì),鄰接表,路徑枚舉,鄰接表路徑枚舉一起來的都見過。每種設(shè)計(jì)都各有優(yōu)劣,如果選擇設(shè)計(jì)依賴于應(yīng)用程序中哪種操作最需要性能上的優(yōu)化。

設(shè)計(jì) 表數(shù)量 查詢直接子 查詢子樹 同時(shí)查詢多個節(jié)點(diǎn)子樹 插入 刪除 移動
鄰接表 1 簡單 需要遞歸 需要遞歸 簡單 簡單 簡單
枚舉路徑 1 簡單 簡單 查多次 相對復(fù)雜 簡單 復(fù)雜
閉包表 2 簡單 簡單 簡單 相對復(fù)雜 簡單 復(fù)雜

綜上所述

  • 只需要建立子父集關(guān)系中可以使用鄰接表方式 ;
  • 涉及向上查找,向下查找的需要建議使用閉包表方式 ;

到此這篇關(guān)于淺談MYSQL中樹形結(jié)構(gòu)表3種設(shè)計(jì)優(yōu)劣分析與分享的文章就介紹到這了,更多相關(guān)MYSQL 樹形結(jié)構(gòu)表內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • MySQL 視圖的基礎(chǔ)操作(五)

    MySQL 視圖的基礎(chǔ)操作(五)

    這篇文章主要為大家詳細(xì)介紹了MySQL 視圖的基礎(chǔ)操作,告訴大家為什么使用視圖?如何創(chuàng)建視圖等問題,感興趣的小伙伴們可以參考一下
    2016-08-08
  • mysql多表join時(shí)候update更新數(shù)據(jù)的方法

    mysql多表join時(shí)候update更新數(shù)據(jù)的方法

    如果item表的name字段為''就用resource_library 表的resource_name字段前面加上字符串Review更新它,他們的關(guān)聯(lián)關(guān)系在表resource_review_link中。
    2011-03-03
  • MySQL學(xué)習(xí)之事務(wù)與并發(fā)控制

    MySQL學(xué)習(xí)之事務(wù)與并發(fā)控制

    這篇文章主要介紹了MySQL中的事務(wù)與并發(fā)控制,一個事務(wù)可以理解為一組操作,這一組操作要么全部執(zhí)行,要么全部不執(zhí)行,想了解更多的小伙伴,可以參考閱讀本文
    2023-03-03
  • SQL中的聯(lián)合索引和普通索引問題

    SQL中的聯(lián)合索引和普通索引問題

    這篇文章主要介紹了SQL中的聯(lián)合索引和普通索引問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • MySQL執(zhí)行update語句和原數(shù)據(jù)相同會再次執(zhí)行嗎

    MySQL執(zhí)行update語句和原數(shù)據(jù)相同會再次執(zhí)行嗎

    這篇文章主要給大家介紹了關(guān)于MySQL執(zhí)行update語句和原數(shù)據(jù)相同是否會再次執(zhí)行的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用MySQL具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-04-04
  • mysql缺少my.ini文件的解決方法

    mysql缺少my.ini文件的解決方法

    使用的這么長時(shí)間的mysql,有一天我突然需要使用mysql?的配置文件my.ini時(shí)發(fā)現(xiàn)沒有這個文件并且這個文件不是被隱藏了,所以本文給大家介紹了mysql缺少my.ini文件的解決方法,需要的朋友可以參考下
    2023-12-12
  • MySQL不就是多表查詢嗎

    MySQL不就是多表查詢嗎

    這篇文章主要介紹了MySQL多表查詢相關(guān)知識,今天我們學(xué)習(xí)要對多張表進(jìn)行相關(guān)操作,相比較于單一的表來說,多張表操作相對復(fù)雜一些,本文給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2023-06-06
  • MySQL?半同步復(fù)制的實(shí)現(xiàn)

    MySQL?半同步復(fù)制的實(shí)現(xiàn)

    半同步復(fù)制是MySQL復(fù)制的一種形式,它結(jié)合了異步復(fù)制和同步復(fù)制的特性,本文主要介紹了?MySQL?半同步復(fù)制的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-09-09
  • MySQL過濾數(shù)據(jù)操作方法梳理

    MySQL過濾數(shù)據(jù)操作方法梳理

    數(shù)據(jù)庫表一般包含大量的數(shù)據(jù),很少需要檢索表中的所有行。通常只是根據(jù)特定的需要提取表數(shù)據(jù)的子集。因此檢索所需數(shù)據(jù)時(shí)需要指定搜索條件,搜索條件也稱為過濾條件
    2022-10-10
  • MySQL 如何查找刪除重復(fù)行

    MySQL 如何查找刪除重復(fù)行

    對MySQL 如何查找刪除重復(fù)行好奇得小伙伴想必也知道MySQL 操作刪除重復(fù)行,難免有時(shí)會需要查看被刪除的重行,今天小編就來給大家詳解,感興趣的小伙伴可以參考下面的文章內(nèi)容
    2021-09-09

最新評論