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

淺談mysql雙層not exists查詢執(zhí)行流程

 更新時(shí)間:2023年06月13日 11:14:20   作者:lanren312  
本文主要介紹了淺談mysql雙層not?exists查詢執(zhí)行流程,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

一、單個(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)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 面試被問select......for update會(huì)鎖表還是鎖行

    面試被問select......for update會(huì)鎖表還是鎖行

    select … for update 是我們常用的對(duì)行加鎖的一種方式,那么select......for update會(huì)鎖表還是鎖行,本文就詳細(xì)的來介紹一下,感興趣的可以了解一下
    2021-11-11
  • MySQL導(dǎo)入sql文件的三種方法小結(jié)

    MySQL導(dǎo)入sql文件的三種方法小結(jié)

    本文主要介紹了MySQL導(dǎo)入sql文件的三種方法小結(jié),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-02-02
  • mysql表的基礎(chǔ)操作匯總(三)

    mysql表的基礎(chǔ)操作匯總(三)

    這篇文章主要匯總了針對(duì)mysql表進(jìn)行的相關(guān)基礎(chǔ)操作,具有一定的實(shí)用性,供大家參考,感興趣的小伙伴們可以參考一下
    2016-08-08
  • MySQL中Union子句不支持order by的解決方法

    MySQL中Union子句不支持order by的解決方法

    這篇文章主要介紹了MySQL中Union子句不支持order by的解決方法,結(jié)合實(shí)例形式分析了在mysql的Union子句中使用order by的方法,需要的朋友可以參考下
    2016-06-06
  • MySQL 不等于的三種使用及區(qū)別

    MySQL 不等于的三種使用及區(qū)別

    MySQL中常用到判斷符號(hào),而不等于是比較常用的符號(hào),不等于主要是三種,本文主要介紹了三種的使用及區(qū)別,感興趣的同學(xué)可以了解一下
    2021-06-06
  • 一文搞懂MySQL索引所有知識(shí)點(diǎn)

    一文搞懂MySQL索引所有知識(shí)點(diǎn)

    這篇文章主要帶你搞懂MySQL索引的所有知識(shí)點(diǎn),我們通常所說的索引,包括聚焦索引、覆蓋索引、組合索引、前綴索引、唯一索引等,沒有特別說明,默認(rèn)都是使用B+樹結(jié)構(gòu)組織,感興趣的小伙伴可以參考閱讀
    2023-03-03
  • 詳解MySQL中concat函數(shù)的用法(連接字符串)

    詳解MySQL中concat函數(shù)的用法(連接字符串)

    本篇文章主要介紹了MySQL中concat函數(shù)的用法(連接字符串),在命令行模式下進(jìn)行測(cè)試。具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。
    2016-12-12
  • mysql事務(wù)隔離級(jí)別詳情

    mysql事務(wù)隔離級(jí)別詳情

    這篇文章主要介紹了mysql事務(wù)隔離級(jí)別,文章圍繞mysql事務(wù)隔離的相關(guān)資料詳細(xì)展開文章重點(diǎn),需要的小伙伴可以參考一下,希望對(duì)你有所幫助
    2021-10-10
  • 批量清除128組節(jié)點(diǎn)db上面過期的binlog釋放磁盤空間實(shí)現(xiàn)思路

    批量清除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上面的binlog
    2013-06-06
  • MySQL數(shù)據(jù)庫(kù)常用命令小結(jié)

    MySQL數(shù)據(jù)庫(kù)常用命令小結(jié)

    這篇文章主要介紹了MySQL數(shù)據(jù)庫(kù)命令,主要包括對(duì)數(shù)據(jù)庫(kù)常用命令及數(shù)據(jù)庫(kù)中對(duì)表的命令,本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2023-01-01

最新評(píng)論