關(guān)于mysql 遞歸查找父節(jié)點(diǎn)的所有子節(jié)點(diǎn)改進(jìn)版

作者:pany

時(shí)間:2019-3-12 21:36

“收獲之前你一定得先付出介时,這是鐵律”

一报咳、背景

在項(xiàng)目中,需要查詢父節(jié)點(diǎn)下所有的子節(jié)點(diǎn)玖详,我的同事是利用遞歸查詢的把介,我發(fā)現(xiàn)他是參考網(wǎng)上的例子寫的。

地址:https://www.cnblogs.com/rainydayfmb/p/8028868.html(其實(shí)這個(gè)作者寫的還算比較仔細(xì))

我在重構(gòu)他的模塊時(shí)蟋座,發(fā)現(xiàn)這個(gè)遞歸查詢并不能將所有的字節(jié)點(diǎn)都查詢出來(lái)拗踢,起初我是懷疑是sql的問(wèn)題,于是按照上面的案例寫了一遍向臀,發(fā)現(xiàn)sql能夠查詢出所有子節(jié)點(diǎn)巢墅,也就證明sql沒(méi)問(wèn)題。于是我懷疑數(shù)據(jù)有問(wèn)題,于是將數(shù)據(jù)仔細(xì)梳理好幾遍砂缩,發(fā)現(xiàn)數(shù)據(jù)也沒(méi)有問(wèn)題作谚。這下我凌亂了,sql 和數(shù)據(jù)都沒(méi)問(wèn)題庵芭,到底是什么造成了這么詭異的事情呢妹懒?

之所以走了這么多彎路,還是自己想圖省事双吆,懶得去分析sql眨唬,走了這么多彎路之后,我不能忍受了好乐,于是我決定好好分析下案例中的sql匾竿。

二、基于第一組數(shù)據(jù)分析(類似于案例中的數(shù)據(jù))


表結(jié)構(gòu):(由于演示蔚万,表結(jié)構(gòu)創(chuàng)建的比較隨意傲胙)

DROP TABLE IF EXISTS `re_menu`;

CREATE TABLE `re_menu`? (

? `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '菜單id',

? `parent_id` int(11) NULL DEFAULT NULL COMMENT '父節(jié)點(diǎn)id',

? `menu_name` varchar(128) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '菜單名稱',

? `menu_url` varchar(128) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT '' COMMENT '菜單路徑',

? `status` tinyint(3) NULL DEFAULT 1 COMMENT '菜單狀態(tài) 1-有效;0-無(wú)效',

? PRIMARY KEY (`id`) USING BTREE

) ENGINE = InnoDB AUTO_INCREMENT = 12223 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Dynamic;

數(shù)據(jù):

INSERT INTO `re_menu`(`id`, `parent_id`, `menu_name`, `menu_url`, `status`) VALUES (1, 0, '菜單1', '', 1);

INSERT INTO `re_menu`(`id`, `parent_id`, `menu_name`, `menu_url`, `status`) VALUES (11, 1, '菜單11', '', 1);

INSERT INTO `re_menu`(`id`, `parent_id`, `menu_name`, `menu_url`, `status`) VALUES (111, 11, '菜單111', '', 1);

INSERT INTO `re_menu`(`id`, `parent_id`, `menu_name`, `menu_url`, `status`) VALUES (122, 11, '菜單122', '', 1);

INSERT INTO `re_menu`(`id`, `parent_id`, `menu_name`, `menu_url`, `status`) VALUES (1111, 111, '菜單12211', '', 1);

INSERT INTO `re_menu`(`id`, `parent_id`, `menu_name`, `menu_url`, `status`) VALUES (1221, 122, '菜單1221', '', 1);

INSERT INTO `re_menu`(`id`, `parent_id`, `menu_name`, `menu_url`, `status`) VALUES (12221, 1221, NULL, '', 1);

INSERT INTO `re_menu`(`id`, `parent_id`, `menu_name`, `menu_url`, `status`) VALUES (12222, 1221, NULL, '', 1);



##? sql 分析:

select

????? t3. id

from (

??????????????select

??????????????????????? t1.id,

???????????? ? ? ? ? ? if ( find_in_set(t1.parent_id, @pids) > 0, @pids:= concat(? @pids, ? ','? , ? t1.id), 0) as ischild

??????????????from

????????????? (

?????????????????????????? select t.id,? t.parent_id from re_menu t where t.status = 1? order by t.parent_id, t.id

????????????? ) t1,

???????????? (select @pids:= '目標(biāo)節(jié)點(diǎn)id' id)? t2

) t3 where t3.ischild != '0'

先關(guān)注 t1 ,它是元數(shù)據(jù)反璃,查詢的是菜單表狀態(tài)是'1'的所有數(shù)據(jù)昵慌,如下圖;


然后再看t2,它是給變量@pids 賦值 '目標(biāo)節(jié)點(diǎn)id'淮蜈;我們?cè)倏?/p>

if ( find_in_set(t1.parent_id, @pids) > 0, @pids:= concat(? @pids, ? ','? , ? t1.id), 0) as ischild

第一條記錄斋攀,判斷 @pids(也就是目標(biāo)節(jié)點(diǎn)的id,我們假如它是1)是否包含第一條記錄的 parent_id(也就是0)梧田,顯然不包括淳蔼,IF(expr1,expr2,expr3)判斷如果expr1 條件不滿足,執(zhí)行expr3裁眯,也就是0鹉梨;

再看第二條記錄,此時(shí)的@pids 仍然是1穿稳,判斷@pids 是否包含第二條記錄的parent_id(也就是1)俯画,顯然包含,那么執(zhí)行expr2司草, @pids:= concat(? @pids, ? ','? , ? t1.id)艰垂,@pids 拼上' ,'再拼上第二條記錄的id,也就是11埋虹,結(jié)果是"1,11 "猜憎,賦值給@pids,后面的依次這樣分析搔课,確實(shí)能把所有的子節(jié)點(diǎn)都查詢出來(lái)胰柑,如下圖:



(注意: = 和 := 的區(qū)別, = 只是在set和update的時(shí)候才是賦值,而 := 除了在set和update時(shí)是賦值柬讨,select 的時(shí)候也是)

三崩瓤、基于第二組數(shù)據(jù)分析


表結(jié)構(gòu):

DROP TABLE IF EXISTS `re_menu_new`;

CREATE TABLE `re_menu_new`? (

? `id` varchar(111) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '菜單id',

? `parent_id` varchar(111) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '父節(jié)點(diǎn)id',

? `menu_name` varchar(128) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '菜單名稱',

? `menu_url` varchar(128) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT '' COMMENT '菜單路徑',

? `status` tinyint(3) NULL DEFAULT 1 COMMENT '菜單狀態(tài) 1-有效;0-無(wú)效',

? PRIMARY KEY (`id`) USING BTREE

) ENGINE = InnoDB CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Dynamic;

數(shù)據(jù):
INSERT INTO `re_menu_new` VALUES ('2001655aa8054fa4841aa192baf45c6b', 'ROOT_FUNCTION_1', '子功能', '', 1);

INSERT INTO `re_menu_new` VALUES ('4cfe206fe97711e88e78fa163e66a663', 'ROOT_FUNCTION_0', '子功能', '', 1);

INSERT INTO `re_menu_new` VALUES ('7a0d9c0d6b164d128dc4e9fa60261169', '4cfe206fe97711e88e78fa163e66a663', '子功能', '', 1);

INSERT INTO `re_menu_new` VALUES ('8cc6b43beae311e88e78fa163e66a663', 'ROOT_FUNCTION_0', '子功能', '', 1);

INSERT INTO `re_menu_new` VALUES ('ROOT', '', '根節(jié)點(diǎn)', '', 1);

INSERT INTO `re_menu_new` VALUES ('ROOT_AREA_MENU', 'ROOT', '菜單節(jié)點(diǎn)', '', 1);

INSERT INTO `re_menu_new` VALUES ('ROOT_FUNCTION_0', 'ROOT_AREA_MENU', '功能1', '', 1);

INSERT INTO `re_menu_new` VALUES ('ROOT_FUNCTION_1', 'ROOT_AREA_MENU', '功能2', '', 1);



##? sql 分析:

select

????? t3. id

from (

??????????????select

??????????????????????? t1.id,

???????????? ? ? ? ? ? if ( find_in_set(t1.parent_id, @pids) > 0, @pids:= concat(? @pids, ? ','? , ?? t1.id), 0) as ischild

?????????????? from

????????????? (

?????????????????????????? select t.id,? t.parent_id from re_menu_new? t where t.status = 1? order by t.parent_id, t.id

????????????? ) t1,

???????????? (select @pids:= 'ROOT' id)? t2

) t3 where t3.ischild != '0'

跟上面一樣踩官,我們先看t1却桶,如下圖;


然后再看t2,它是給變量@pids 賦值 'ROOT'蔗牡;我們?cè)倏?/p>

if ( find_in_set(t1.parent_id, @pids) > 0, @pids:= concat(? @pids, ? ','? , ? t1.id), 0) as ischild

第一條記錄颖系,判斷 @pids(ROOT)是否包含第一條記錄的 parent_id(無(wú)),顯然不包括辩越,IF(expr1,expr2,expr3)判斷如果expr1 條件不滿足嘁扼,執(zhí)行expr3,也就是0黔攒;

再看第二條記錄趁啸,此時(shí)的@pids 仍然是ROOT,判斷@pids 是否包含第二條記錄的parent_id(也就是4cfe206fe97711e88e78fa163e66a663)督惰,顯然不包含不傅,那么執(zhí)行expr3, 返回0姑丑,而 where t3.ischild != '0'? 這里看到 0是被排除的,被認(rèn)為不是子節(jié)點(diǎn)辞友,但是從關(guān)系可以看出id是7a0d9c0d6b164d128dc4e9fa60261169 的節(jié)點(diǎn)很明顯是ROOT下的葉子節(jié)點(diǎn)栅哀。

問(wèn)題在哪呢?

由于 執(zhí)行 select t.id,? t.parent_id from re_menu_new? t where t.status = 1? order by t.parent_id, t.id 的排序?qū)е铝藛?wèn)題称龙,因?yàn)榕判蛑罅羰埃覀僷arent_id的順序不在是從ROOT依次向下排列了,而是中間多了一個(gè)4cfe206fe97711e88e78fa163e66a663鲫尊。所以這個(gè)sql的成敗關(guān)鍵在于元數(shù)據(jù)的排序痴柔。如果我們將第一組數(shù)據(jù)的元素順序改成倒序,如下:

select t.id, t.parent_id from re_menu t where t.status = 1 order by t.parent_id DESC


我們?cè)賵?zhí)行查詢子節(jié)點(diǎn)的sql疫向,也會(huì)出錯(cuò)咳蔚,如下:



我總結(jié)這么多,其實(shí)就像說(shuō)明這個(gè)sql的成敗關(guān)鍵搔驼,如果你的id和pid的關(guān)系是序列這種谈火,使用上面的sql完全沒(méi)問(wèn)題,但是如果類似第二種數(shù)據(jù)舌涨,可能就不適合了糯耍。我說(shuō)的可能有點(diǎn)啰嗦,非常抱歉,因?yàn)槲乙郧爱?dāng)過(guò)老師温技,喜歡講東西多舉例子革为。



那遇到這類問(wèn)題是不是沒(méi)有解決方案了呢?并不是舵鳞,還有其他的寫法震檩,網(wǎng)上也有很多例子。

四系任、解決方案

其實(shí)還有其他的方案恳蹲,主要我們根據(jù)具體環(huán)境去用,有些環(huán)境下俩滥,我們是不能隨便添加存儲(chǔ)過(guò)程的嘉蕾。

下面的解決方案是利用函數(shù)的:(注意函數(shù)的創(chuàng)建用戶要根據(jù)具體的改下,DEFINER=`root`@`%???? 這個(gè)root是指定的用戶霜旧,小白要注意)

DELIMITER //

CREATE DEFINER=`root`@`%` FUNCTION `getChildLst`(rootId varchar(50)) RETURNS varchar(1000) CHARSET utf8

BEGIN

DECLARE sTemp VARCHAR(1000);

DECLARE sTempChd VARCHAR(1000);

SET sTemp = '$';

SET sTempChd =rootId;

WHILE sTempChd is not null DO

SET sTemp = concat(sTemp,',',sTempChd);

SELECT group_concat(id) INTO sTempChd FROM re_menu_new where parent_id<>id and FIND_IN_SET(parent_id,sTempChd)>0;

END WHILE;

RETURN sTemp;

END//

創(chuàng)建完成之后错忱,使用下面的查詢:

select

? ? r. id,r.parent_id

from re_menu_new r

where FIND_IN_SET(r.id,(select `getChildLst`('ROOT') as id))


所有的結(jié)果都查詢出來(lái)了。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末挂据,一起剝皮案震驚了整個(gè)濱河市以清,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌崎逃,老刑警劉巖掷倔,帶你破解...
    沈念sama閱讀 218,755評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異个绍,居然都是意外死亡勒葱,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門巴柿,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)凛虽,“玉大人,你說(shuō)我怎么就攤上這事广恢】” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 165,138評(píng)論 0 355
  • 文/不壞的土叔 我叫張陵钉迷,是天一觀的道長(zhǎng)至非。 經(jīng)常有香客問(wèn)我,道長(zhǎng)糠聪,這世上最難降的妖魔是什么睡蟋? 我笑而不...
    開(kāi)封第一講書人閱讀 58,791評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮枷颊,結(jié)果婚禮上戳杀,老公的妹妹穿的比我還像新娘该面。我一直安慰自己,他們只是感情好信卡,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,794評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布隔缀。 她就那樣靜靜地躺著,像睡著了一般傍菇。 火紅的嫁衣襯著肌膚如雪猾瘸。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書人閱讀 51,631評(píng)論 1 305
  • 那天丢习,我揣著相機(jī)與錄音牵触,去河邊找鬼。 笑死咐低,一個(gè)胖子當(dāng)著我的面吹牛揽思,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播见擦,決...
    沈念sama閱讀 40,362評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼钉汗,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了鲤屡?” 一聲冷哼從身側(cè)響起损痰,我...
    開(kāi)封第一講書人閱讀 39,264評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎酒来,沒(méi)想到半個(gè)月后卢未,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,724評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡堰汉,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,900評(píng)論 3 336
  • 正文 我和宋清朗相戀三年辽社,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片衡奥。...
    茶點(diǎn)故事閱讀 40,040評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡爹袁,死狀恐怖远荠,靈堂內(nèi)的尸體忽然破棺而出矮固,到底是詐尸還是另有隱情,我是刑警寧澤譬淳,帶...
    沈念sama閱讀 35,742評(píng)論 5 346
  • 正文 年R本政府宣布档址,位于F島的核電站,受9級(jí)特大地震影響邻梆,放射性物質(zhì)發(fā)生泄漏守伸。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,364評(píng)論 3 330
  • 文/蒙蒙 一浦妄、第九天 我趴在偏房一處隱蔽的房頂上張望尼摹。 院中可真熱鬧见芹,春花似錦、人聲如沸蠢涝。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 31,944評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)和二。三九已至徘铝,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間惯吕,已是汗流浹背惕它。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 33,060評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留废登,地道東北人淹魄。 一個(gè)月前我還...
    沈念sama閱讀 48,247評(píng)論 3 371
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像钳宪,于是被迫代替她去往敵國(guó)和親揭北。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,979評(píng)論 2 355