淺談mysql雙層not exists查詢執(zhí)行流程
一、單個EXISTS、NOT EXISTS用法
DROP TABLE IF EXISTS `t_staff`; CREATE TABLE `t_staff` ( `id` int(11) NOT NULL, `name` varchar(255) DEFAULT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='人員表'; DROP TABLE IF EXISTS `t_major`; CREATE TABLE `t_major` ( `id` int(11) NOT NULL, `staff_id` int(11) NOT NULL, `name` varchar(255) NOT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='技能表'; INSERT INTO `t_staff` VALUES ('1', '張遼'); INSERT INTO `t_staff` VALUES ('2', '趙云'); INSERT INTO `t_staff` VALUES ('3', '夏侯淵'); INSERT INTO `t_major` VALUES ('1', '2', '大鵬展翅'); INSERT INTO `t_major` VALUES ('2', '2', '無敵風(fēng)火輪'); INSERT INTO `t_major` VALUES ('3', '3', '橫掃千軍');
EXISTS內(nèi)層查詢【非空】外層的where返回【真值】;內(nèi)層查詢【為空】外層的where返回【假值】
NOT EXISTS內(nèi)層查詢【非空】外層的where返回【假值】;內(nèi)層查詢【為空】外層的where返回【真值】
SELECT * FROM t_student_course WHERE TRUE; -- 能查出數(shù)據(jù) SELECT * FROM t_student_course WHERE FALSE; -- 不能查出數(shù)據(jù)
1、查詢至少有一個技能的人員信息(in、EXISTS)
SELECT * FROM t_staff where id in (SELECT staff_id FROM t_major); SELECT * FROM t_staff where EXISTS ( SELECT 1 from t_major where t_staff.id = t_major.staff_id ); -- 趙云、夏侯淵
2、查詢一個技能都沒有的人員信息(in、EXISTS)
SELECT * FROM t_staff where id not in (SELECT staff_id FROM t_major); SELECT * FROM t_staff where NOT EXISTS ( SELECT 1 from t_major where t_staff.id = t_major.staff_id ); -- 張遼
二、EXISTS、NOT EXISTS(3張表)
DROP TABLE IF EXISTS `t_student`; CREATE TABLE `t_student` ( `id` int(11) NOT NULL, `name` varchar(255) DEFAULT NULL COMMENT '姓名', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; DROP TABLE IF EXISTS `t_course`; CREATE TABLE `t_course` ( `id` int(11) NOT NULL, `name` varchar(255) DEFAULT NULL COMMENT '課程名稱', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; DROP TABLE IF EXISTS `t_student_course`; CREATE TABLE `t_student_course` ( `id` int(11) NOT NULL, `sid` varchar(255) NOT NULL, `cid` varchar(255) NOT NULL, `score` decimal(10,0) DEFAULT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; INSERT INTO `t_student` VALUES (1, '趙子龍'); INSERT INTO `t_student` VALUES (2, '關(guān)羽'); INSERT INTO `t_student` VALUES (3, '張飛'); INSERT INTO `t_student` VALUES (4, '黃忠'); INSERT INTO `t_student` VALUES (5, '馬超'); INSERT INTO `t_course` VALUES (1, '語文'); INSERT INTO `t_course` VALUES (2, '數(shù)學(xué)'); INSERT INTO `t_course` VALUES (3, '英語'); INSERT INTO `t_student_course` VALUES (1, 1, 1, 95); INSERT INTO `t_student_course` VALUES (2, 1, 2, 99); INSERT INTO `t_student_course` VALUES (3, 1, 3, 100); INSERT INTO `t_student_course` VALUES (4, 2, 1, 99); INSERT INTO `t_student_course` VALUES (5, 2, 2, 100); INSERT INTO `t_student_course` VALUES (6, 3, 2, 100); INSERT INTO `t_student_course` VALUES (7, 3, 3, 95); INSERT INTO `t_student_course` VALUES (8, 4, 1, 100);
EXISTS和NOT EXISTS相關(guān)子查詢什么時候返回,返回有兩個條件(很重要)。
1、子查詢找到一個匹配的立即返回;
2、子查詢遍歷所有,沒有找到一個匹配的返回為空。
1、查詢出選修了全部課程的學(xué)生姓名
SELECT name from t_student where id in( SELECT sid FROM t_student_course GROUP BY sid HAVING (count(sid)) = (SELECT count(*) from t_course) ) -- 記錄一般寫法 SELECT name from t_student s where NOT EXISTS ( SELECT 1 from t_course c where NOT EXISTS ( SELECT 1 from t_student_course sc where sc.sid = s.id and sc.cid = c.id ) ) -- 趙子龍
子查詢只會返回true、false,select后面寫1就行了。將查詢語句定義成三個變量,后面好解釋。
let a1 = SELECT 1 from t_student_course sc where sc.sid = s.id and sc.cid = c.id let b1 = SELECT 1 FROM t_role WHERE NOT EXISTS (a1) let c1 = SELECT * FROM t_account WHERE NOT EXISTS (b1);
1 1 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
1 2 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
1 3 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
一個都沒找就返回空,最外層的NOT EXISTS就為真,輸出2 1 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
2 2 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
2 3 遍歷sc表沒有找到一個匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出3 1 遍歷sc表沒有找到一個匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出4 1 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
4 2 遍歷sc表沒有找到一個匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出5 1 遍歷sc表沒有找到一個匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出
2、查詢至少選修了一門課程的學(xué)生
SELECT name from t_student s where EXISTS ( SELECT 1 from t_course c where EXISTS ( SELECT 1 from t_student_course sc where sc.sid = s.id and sc.cid = c.id ) ) -- 趙子龍、關(guān)羽、張飛、黃忠
1 1 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的EXISTS就為真,輸出2 1 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的EXISTS就為真,輸出3 1 遍歷sc表沒有找到一個匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回
3 2 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的EXISTS就為真,輸出4 1 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的EXISTS就為真,輸出5 1 遍歷sc表沒有找到一個匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回
5 2 遍歷sc表沒有找到一個匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回
5 3 遍歷sc表沒有找到一個匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回一個都沒找就返回空,最外層的EXISTS就為假,不輸出
3、查詢沒有選擇所有課程的學(xué)生
SELECT name from t_student s where EXISTS ( SELECT 1 from t_course c where NOT EXISTS ( SELECT 1 from t_student_course sc where sc.sid = s.id and sc.cid = c.id ) ) -- 關(guān)羽、張飛、黃忠、馬超
1 1 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
1 2 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
1 3 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
一個都沒找就返回空,最外層的EXISTS就為假,不輸出2 1 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
2 2 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
2 3 遍歷sc表沒有找到一個匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的EXISTS就為真,輸出3 1 遍歷sc表沒有找到一個匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的EXISTS就為真,輸出4 1 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
4 2 遍歷sc表沒有找到一個匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的EXISTS就為真,輸出5 1 遍歷sc表沒有找到一個匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的EXISTS就為假,不輸出
4、查詢一門課也沒有選的學(xué)生
SELECT name FROM t_student s where NOT EXISTS ( SELECT 1 from t_course c where EXISTS ( SELECT 1 from t_student_course sc where sc.sid = s.id and sc.cid = c.id ) ) -- 馬超
1 1 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出2 1 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出3 1 遍歷sc表沒有找到一個匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回
3 2 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出4 1 遍歷sc表找到一個匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出5 1 遍歷sc表沒有找到一個匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回
5 2 遍歷sc表沒有找到一個匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回
5 3 遍歷sc表沒有找到一個匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回一個都沒找就返回空,最外層的NOT EXISTS就為真,輸出
5、查詢至少選修了學(xué)生2選修的全部課程的學(xué)生名單
select * from t_student s where id != 2 and NOT EXISTS ( select 1 from t_student_course sc where sid = 2 and NOT EXISTS ( select 1 from t_student_course sc2 where sc2.cid = sc.cid and sc2.sid = s.id ) ) -- 趙子龍
6、選出每門課程中成績最高的學(xué)生
SELECT * FROM t_student_course sc where sc.score = (SELECT max(score) FROM t_student_course sc2 where sc2.cid = sc.cid) SELECT * FROM t_student_course sc where NOT EXISTS ( SELECT 1 FROM t_student_course sc2 where sc2.cid = sc.cid and sc2.score > sc.score )
id sid cid score
3 1 3 100
4 1 4 97
7 2 2 100
8 2 3 100
9 3 1 100
11 4 5 100
最后來說說EXISTS、IN用法
EXISTS查詢:先執(zhí)行一次外部查詢,然后為外部查詢返回的每一行執(zhí)行一次子查詢,如果外部查詢返回100行記錄,sql就將執(zhí)行101次查詢。
IN查詢:先查詢子查詢,然后把子查詢的結(jié)果放到外部查詢中進(jìn)行查詢。IN語句在mysql中沒有參數(shù)個數(shù)的限制,但是mysql中sql語句有長度大小限制,整段最大為4M。IN引導(dǎo)的子查詢只能返回一個字段。
EXISTS、IN怎么用
當(dāng)子查詢的表大的時候,使用EXISTS可以有效減少總的循環(huán)次數(shù)來提升速度,當(dāng)外查詢的表大的時候,使用IN可以有效減少對外查詢表循環(huán)遍歷來提升速度,
顯然,外表大而子表小時,IN的效率更高,而外表小,子表大時,EXISTS的效率更高,若兩表差不多大,則差不多。
到此這篇關(guān)于淺談mysql雙層not exists查詢執(zhí)行流程的文章就介紹到這了,更多相關(guān)mysql雙層not exists內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MySQL 那些常見的錯誤設(shè)計規(guī)范,你都知道嗎
今天來看一看 MySQL 設(shè)計規(guī)范中幾個常見的錯誤例子,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧2021-07-07mysql+shardingSphere的分庫分表實現(xiàn)示例
分庫分表是一種場景解決方案,它的出現(xiàn)是為了解決一些場景問題的,本文主要介紹了mysql+shardingSphere的分庫分表實現(xiàn)示例,具有一定的參考價值,感興趣的可以2024-04-04解決創(chuàng)建主鍵報錯:Incorrect column specifier for
這篇文章主要介紹了解決創(chuàng)建主鍵報錯:Incorrect column specifier for column‘id‘問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-08-08mysql主從基于docker和django實現(xiàn)讀寫分離
這篇文章主要介紹了mysql主從基于docker和django實現(xiàn)讀寫分離,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價值,感興趣的小伙伴可以參考一下2022-08-08