淺談mysql雙層not exists查詢執(zhí)行流程
一、單個(gè)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、查詢至少有一個(gè)技能的人員信息(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、查詢一個(gè)技能都沒有的人員信息(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)子查詢什么時(shí)候返回,返回有兩個(gè)條件(很重要)。
1、子查詢找到一個(gè)匹配的立即返回;
2、子查詢遍歷所有,沒有找到一個(gè)匹配的返回為空。
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 ) ) -- 趙子龍
子查詢只會(huì)返回true、false,select后面寫1就行了。將查詢語句定義成三個(gè)變量,后面好解釋。
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表找到一個(gè)匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
1 2 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
1 3 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
一個(gè)都沒找就返回空,最外層的NOT EXISTS就為真,輸出2 1 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
2 2 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
2 3 遍歷sc表沒有找到一個(gè)匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出3 1 遍歷sc表沒有找到一個(gè)匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出4 1 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
4 2 遍歷sc表沒有找到一個(gè)匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出5 1 遍歷sc表沒有找到一個(gè)匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的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表找到一個(gè)匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的EXISTS就為真,輸出2 1 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的EXISTS就為真,輸出3 1 遍歷sc表沒有找到一個(gè)匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回
3 2 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的EXISTS就為真,輸出4 1 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的EXISTS就為真,輸出5 1 遍歷sc表沒有找到一個(gè)匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回
5 2 遍歷sc表沒有找到一個(gè)匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回
5 3 遍歷sc表沒有找到一個(gè)匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回一個(gè)都沒找就返回空,最外層的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表找到一個(gè)匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
1 2 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
1 3 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
一個(gè)都沒找就返回空,最外層的EXISTS就為假,不輸出2 1 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
2 2 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
2 3 遍歷sc表沒有找到一個(gè)匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的EXISTS就為真,輸出3 1 遍歷sc表沒有找到一個(gè)匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的EXISTS就為真,輸出4 1 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的NOT EXISTS就為假,b1為空,不能返回
4 2 遍歷sc表沒有找到一個(gè)匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的EXISTS就為真,輸出5 1 遍歷sc表沒有找到一個(gè)匹配的返回空(假),最內(nèi)層的NOT EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的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表找到一個(gè)匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出2 1 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出3 1 遍歷sc表沒有找到一個(gè)匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回
3 2 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出4 1 遍歷sc表找到一個(gè)匹配的立即返回真,最內(nèi)層的EXISTS就為真,b1不為空,立即返回
找到一個(gè)匹配的立即返回真,最外層的NOT EXISTS就為假,不輸出5 1 遍歷sc表沒有找到一個(gè)匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回
5 2 遍歷sc表沒有找到一個(gè)匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回
5 3 遍歷sc表沒有找到一個(gè)匹配的返回空,最內(nèi)層的EXISTS就為假,b1為空,不能返回一個(gè)都沒找就返回空,最外層的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、選出每門課程中成績(jī)最高的學(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ù)個(gè)數(shù)的限制,但是mysql中sql語句有長(zhǎng)度大小限制,整段最大為4M。IN引導(dǎo)的子查詢只能返回一個(gè)字段。
EXISTS、IN怎么用
當(dāng)子查詢的表大的時(shí)候,使用EXISTS可以有效減少總的循環(huán)次數(shù)來提升速度,當(dāng)外查詢的表大的時(shí)候,使用IN可以有效減少對(duì)外查詢表循環(huán)遍歷來提升速度,
顯然,外表大而子表小時(shí),IN的效率更高,而外表小,子表大時(shí),EXISTS的效率更高,若兩表差不多大,則差不多。
到此這篇關(guān)于淺談mysql雙層not exists查詢執(zhí)行流程的文章就介紹到這了,更多相關(guān)mysql雙層not exists內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Mysql中in和exists的區(qū)別?&?not?in、not?exists、left?join的相互轉(zhuǎn)換問題
- mysql exists與not exists實(shí)例詳解
- mysql not in、left join、IS NULL、NOT EXISTS 效率問題記錄
- UCenter info: MySQL Query Error SQL:SELECT value FROM [Table]vars WHERE noteexists
- mysql insert if not exists防止插入重復(fù)記錄的方法
- MySQL: mysql is not running but lock exists 的解決方法
- 淺談mysql的not exists走不走索引
相關(guān)文章
面試被問select......for update會(huì)鎖表還是鎖行
select … for update 是我們常用的對(duì)行加鎖的一種方式,那么select......for update會(huì)鎖表還是鎖行,本文就詳細(xì)的來介紹一下,感興趣的可以了解一下2021-11-11詳解MySQL中concat函數(shù)的用法(連接字符串)
本篇文章主要介紹了MySQL中concat函數(shù)的用法(連接字符串),在命令行模式下進(jìn)行測(cè)試。具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。2016-12-12批量清除128組節(jié)點(diǎn)db上面過期的binlog釋放磁盤空間實(shí)現(xiàn)思路
在 一臺(tái)db跳轉(zhuǎn)機(jī)上面, 寫一個(gè)腳本,訪問slave,遠(yuǎn)程獲取正在復(fù)制的master上面的binlog位置, 然后再遠(yuǎn)程去purge master上面的binlog2013-06-06MySQL數(shù)據(jù)庫(kù)常用命令小結(jié)
這篇文章主要介紹了MySQL數(shù)據(jù)庫(kù)命令,主要包括對(duì)數(shù)據(jù)庫(kù)常用命令及數(shù)據(jù)庫(kù)中對(duì)表的命令,本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2023-01-01