Mysql exists用法小結(jié)

EXISTS用于檢查子查詢是否至少會(huì)返回一行數(shù)據(jù),該子查詢實(shí)際上并不返回任何數(shù)據(jù),而是返回值True或False。

EXISTS 指定一個(gè)子查詢,檢測(cè)行的存在。語法:EXISTS subquery。參數(shù) subquery 是一個(gè)受限的 SELECT 語句 (不允許有 COMPUTE 子句和 INTO 關(guān)鍵字)。結(jié)果類型為 Boolean,如果子查詢包含行,則返回 TRUE。文章源自四五設(shè)計(jì)網(wǎng)-http://www.133122.cn/44069.html

在mysql中,exists用于檢查子查詢是否至少會(huì)返回一行數(shù)據(jù),該子查詢實(shí)際上并不返回任何數(shù)據(jù),而是返回true或false,語法為“SELECT 字段 FROM table WHERE EXISTS (subquery);”。文章源自四五設(shè)計(jì)網(wǎng)-http://www.133122.cn/44069.html

語法:

SELECT 字段 FROM table WHERE EXISTS (subquery);文章源自四五設(shè)計(jì)網(wǎng)-http://www.133122.cn/44069.html

參數(shù):

subquery是一個(gè)受限的SELECT語句(不允許有COMPUTE子句和INTO關(guān)鍵字)文章源自四五設(shè)計(jì)網(wǎng)-http://www.133122.cn/44069.html

示例:

SELECT * FROM A WHERE EXISTS (SELECT 1 FROM B WHERE B.id = A.id);文章源自四五設(shè)計(jì)網(wǎng)-http://www.133122.cn/44069.html

EXISTS執(zhí)行順序:文章源自四五設(shè)計(jì)網(wǎng)-http://www.133122.cn/44069.html

1、首先執(zhí)行一次外部查詢,并緩存結(jié)果集,如 SELECT * FROM A文章源自四五設(shè)計(jì)網(wǎng)-http://www.133122.cn/44069.html

2、遍歷外部查詢結(jié)果集的每一行記錄R,代入子查詢中作為條件進(jìn)行查詢,如 SELECT 1 FROM B WHERE B.id = A.id文章源自四五設(shè)計(jì)網(wǎng)-http://www.133122.cn/44069.html

3、如果子查詢有返回結(jié)果,則EXISTS子句返回TRUE,這一行R可作為外部查詢的結(jié)果行,否則不能作為結(jié)果文章源自四五設(shè)計(jì)網(wǎng)-http://www.133122.cn/44069.html

示例如下:文章源自四五設(shè)計(jì)網(wǎng)-http://www.133122.cn/44069.html

假設(shè)現(xiàn)在有三張表:

student:學(xué)生表,其中有字段sno為學(xué)號(hào),sname為學(xué)生姓名

course:課程表,其中有字段cno為課程號(hào),cname為課程名稱

student_course_relation:選課表,記錄學(xué)生選擇了哪些課程,其中有字段sno為學(xué)號(hào),cno為課程號(hào)

下面通過幾個(gè)示例來說明一下EXISTS和NOT EXISTS的用法,及其與IN和NOT IN的區(qū)別

1、在子查詢中使用NULL,仍然返回結(jié)果集

下面三種情況返回?cái)?shù)據(jù)相同,都會(huì)返回student表的所有數(shù)據(jù):

1
2
3
select * from student;?
select * from student where exists (select 1);?
select * from student where exists (select null);

2、EXISTS子查詢返回的是一個(gè)布爾值true或false

EXISTS用于檢查子查詢是否至少會(huì)返回一行數(shù)據(jù),該子查詢實(shí)際上并不返回任何數(shù)據(jù),而是返回布爾值true或false,EXISTS指定一個(gè)子查詢,檢測(cè)行的存在。

EXISTS只在乎子查詢中是否有記錄,與具體的結(jié)果集無關(guān),所以下面示例中,子查詢中的select sno也可以換成select cno或者select 1,查詢出的結(jié)果集是一樣的。

查詢所有選修了課程號(hào)為3的學(xué)生:

1
2
3
4
5
6
select * from student a?
where exists (select sno from student_course_relation b where b.cno=3 and b.sno=a.sno);?
select * from student a?
where exists (select cno from student_course_relation b where b.cno=3 and b.sno=a.sno);?
select * from student a?
where exists (select 1 from student_course_relation b where b.cno=3 and b.sno=a.sno);

補(bǔ)充示例

一張活動(dòng)配置主表activity_main,通過act_code來唯一標(biāo)明一場(chǎng)活動(dòng),活動(dòng)舉辦地點(diǎn)適配表activity_area,通過act_code與主表進(jìn)行關(guān)聯(lián),活動(dòng)獎(jiǎng)品表activity_sku,通過act_code與主表進(jìn)行關(guān)聯(lián)。

活動(dòng)主表

1
2
3
4
5
6
7
CREATE TABLE `activity_main` (
`id` bigint(20) NOT NULL AUTO_INCREMENT,
`act_code` varchar(255) NOT NULL COMMENT '活動(dòng)代碼',
`act_name` varchar(255) NOT NULL COMMENT '活動(dòng)名稱',
PRIMARY KEY (`id`),
UNIQUE KEY `uniq_code` (`act_code`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='活動(dòng)主表'

活動(dòng)在哪些網(wǎng)站舉辦的適配表

1
2
3
4
5
6
CREATE TABLE `activity_area` (
?`id` bigint(20) NOT NULL AUTO_INCREMENT,
?`act_code` varchar(255) NOT NULL COMMENT '活動(dòng)代碼',
?`area` varchar(255) NOT NULL COMMENT '參與此活動(dòng)的網(wǎng)站',
?PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='活動(dòng)適配的網(wǎng)站列表'

活動(dòng)獎(jiǎng)品表

1
2
3
4
5
6
CREATE TABLE `activity_sku` (
?`id` bigint(20) NOT NULL AUTO_INCREMENT,
?`act_code` varchar(255) NOT NULL COMMENT '活動(dòng)代碼',
?`sku` varchar(255) NOT NULL COMMENT '活動(dòng)贈(zèng)送的商品',
?PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='活動(dòng)贈(zèng)品表'

比較使用 EXISTS 和 IN 的查詢

這個(gè)例子比較了兩個(gè)語義類似的查詢。第一個(gè)查詢使用 IN 而第二個(gè)查詢使用 EXISTS。注意兩個(gè)查詢返回相同的信息。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
# 查詢體重秤
select * from activity_main where act_code in (
select act_code from activity_sku where sku = '翎野君的體脂稱'
)
# 查詢體重秤
select * from activity_main a where exists (
select 1 from activity_sku b where a.act_code = b.act_code and b.sku = '翎野君的體脂稱'
)
# 模糊查詢B-BEKO英國(guó)嬰兒推車
select * from activity_main where act_code in (
select act_code from activity_sku where sku like '%B-BEKO%'
)
# 模糊查詢B-BEKO英國(guó)嬰兒推車
select * from activity_main a where exists (
select 1 from activity_sku b where a.act_code = b.act_code and b.sku like '%B-BEKO%'
)
# 查詢?cè)诓┛蛨@舉辦的活動(dòng)
select * from activity_main where act_code in (
select act_code from activity_area where area = '博客園'
)
# 查詢?cè)诓┛蛨@舉辦的活動(dòng)
select * from activity_main a where exists (
select 1 from activity_area b where a.act_code = b.act_code and b.area = '博客園'
)
# 在博客園舉辦活動(dòng)且活動(dòng)獎(jiǎng)品為華為手機(jī)的活動(dòng)信息
select * from activity_main where act_code in (
select act_code from activity_area where area = '博客園' and act_code in (
select act_code from activity_sku where sku = '華為P30Pro'
))
# 內(nèi)層的exists語句只在當(dāng)前where語句中生效,最終是否返回,要根據(jù)最外層的exists判斷,如果是 true(真)就返回到結(jié)果集,為 false(假)丟棄。
select * from activity_main a where exists (
select 1 from activity_area b where a.act_code = b.act_code and b.area = '博客園' and exists
(select 1 from activity_sku c where a.act_code = c.act_code and c.sku = '華為P30Pro')
)

以上就是Mysql exists用法小結(jié)的詳細(xì)內(nèi)容

繼續(xù)閱讀
我的微信
微信掃一掃
weinxin
我的微信
惠生活福利社
微信掃一掃
weinxin
我的公眾號(hào)
 

發(fā)表評(píng)論

匿名網(wǎng)友
:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

拖動(dòng)滑塊以完成驗(yàn)證