成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

MySQL查詢語句中的IN和Exists對比分析

數(shù)據(jù)庫 MySQL
最近在寫SQL語句時,對選擇IN 還是Exists 猶豫不決,于是把兩種方法的SQL都寫出來對比一下執(zhí)行效率,發(fā)現(xiàn)IN的查詢效率比Exists高了很多,于是想當然的認為IN的效率比Exists好,但本著尋根究底的原則,我想知道這個結論是否適用所有場景,以及為什么會出現(xiàn)這個結果。

 

MySQL查詢語句中的IN和Exists對比分析

背景介紹

最近在寫SQL語句時,對選擇IN 還是Exists 猶豫不決,于是把兩種方法的SQL都寫出來對比一下執(zhí)行效率,發(fā)現(xiàn)IN的查詢效率比Exists高了很多,于是想當然的認為IN的效率比Exists好,但本著尋根究底的原則,我想知道這個結論是否適用所有場景,以及為什么會出現(xiàn)這個結果。

網(wǎng)上查了一下相關資料,大體可以歸納為:外部表小,內部表大時,適用Exists;外部表大,內部表小時,適用IN。那我就困惑了,因為我的SQL語句里面,外表只有1W級別的數(shù)據(jù),內表有30W級別的數(shù)據(jù),按網(wǎng)上的說法應該是Exists的效率會比IN高的,但我的結果剛好相反!!

“沒有調查就沒有發(fā)言權”!于是我開始研究IN 和Exists的實際執(zhí)行過程,從實踐的角度出發(fā),在根本上去尋找原因,于是有了這篇博文分享。

實驗數(shù)據(jù)

我的實驗數(shù)據(jù)包括兩張表:t_author表 和 t_poetry表。

對應表的數(shù)據(jù)量:

t_author表,13355條記錄;

t_poetry表,289917條記錄。

對應的表結構如下:

 

  1. CREATE TABLE t_poetry (  
  2. id bigint(20) NOT NULL AUTO_INCREMENT,  
  3. poetry_id bigint(20) NOT NULL COMMENT '詩詞id' 
  4. poetry_name varchar(200) NOT NULL COMMENT '詩詞名稱' 
  5. <font color=red> author_id bigint(20) NOT NULL COMMENT '作者id'</font>  
  6. PRIMARY KEY (id),  
  7. <font color=red>  
  8. UNIQUE KEY pid_idx (poetry_id) USING BTREE,  
  9. KEY aid_idx (author_id) USING BTREE</font>  
  10. ) ENGINE=InnoDB AUTO_INCREMENT=291270 DEFAULT CHARSET=utf8mb4  
  11. CREATE TABLE t_author (  
  12. id int(15) NOT NULL AUTO_INCREMENT,  
  13. <font color=red> author_id bigint(20) NOT NULL,</font>  
  14. author_name varchar(32) NOT NULL 
  15. dynasty varchar(16) NOT NULL 
  16. poetry_num int(8) NOT NULL DEFAULT '0'  
  17. PRIMARY KEY (id),  
  18. <font color=red>UNIQUE KEY authorid_idx (author_id) USING BTREE</font>  
  19. ) ENGINE=InnoDB AUTO_INCREMENT=13339 DEFAULT CHARSET=utf8mb4 

 

執(zhí)行計劃分析

IN 執(zhí)行過程

sql示例:

 

  1. select * from tabA where tabA.x in (select x from tabB where y>0 ); 

其執(zhí)行計劃:

(1)執(zhí)行tabB表的子查詢,得到結果集B,可以使用到tabB表的索引y;

(2)執(zhí)行tabA表的查詢,查詢條件是tabA.x在結果集B里面,可以使用到tabA表的索引x。

Exists執(zhí)行過程

sql示例:

 

  1. select from tabA where exists (select from tabB where y>0); 

其執(zhí)行計劃:

(1)先將tabA表所有記錄取到。

(2)逐行針對tabA表的記錄,去關聯(lián)tabB表,判斷tabB表的子查詢是否有返回數(shù)據(jù),5.5之后的版本使用Block Nested Loop(Block 嵌套循環(huán))。

(3)如果子查詢有返回數(shù)據(jù),則將tabA當前記錄返回到結果集。

tabA相當于取全表數(shù)據(jù)遍歷,tabB可以使用到索引。

實驗過程

實驗針對相同結果集的IN和Exists 的SQL語句進行分析。

包含IN的SQL語句:

 

  1. select from t_author ta where author_id in  
  2. (select author_id from t_poetry tp where tp.poetry_id>3650 ); 

 

包含Exists的SQL語句:

 

  1. select from t_author ta where exists  
  2. (select * from t_poetry tp where tp.poetry_id>3650 and tp.author_id=ta.author_id); 

 

第一次實驗

數(shù)據(jù)情況

t_author表,13355條記錄;t_poetry表,子查詢篩選結果集 where poetry_id>293650 ,121條記錄;

執(zhí)行結果

使用exists耗時0.94S, 使用in耗時0.03S,<font color=red>IN 效率高于Exists</font>。

原因分析

對t_poetry表的子查詢結果集很小,且兩者在t_poetry表都能使用索引,對t_poetry子查詢的消耗基本一致。兩者區(qū)別在于,使用 in 時,t_author表能使用索引:

MySQL查詢語句中的IN 和Exists 對比分析

 

使用exists時,t_author表全表掃描:

MySQL查詢語句中的IN 和Exists 對比分析

 

在子查詢結果集較小時,查詢耗時主要表現(xiàn)在對t_author表的遍歷上。

第二次實驗

數(shù)據(jù)情況

t_author表,13355條記錄;t_poetry表,子查詢篩選結果集 where poetry_id>3650 ,287838條記錄;

執(zhí)行時間

使用exists耗時0.12S, 使用in耗時0.48S,<font color=red>Exists IN</font>。

原因分析

兩者的索引使用情況跟第一次實驗是一致的,唯一區(qū)別是子查詢篩選結果集的大小不同,但實驗結果已經(jīng)跟第一次的不同了。這種情況下子查詢結果集很大,我們看看mysql的查詢計劃:

使用in時,由于子查詢結果集很大,對t_author和t_poetry表都接近于全表掃描,此時對t_author表的遍歷耗時差異對整體效率影響可以忽略,執(zhí)行計劃里多了一行<auto_key>,在接近全表掃描的情況下,mysql優(yōu)化器選擇了auto_key來遍歷t_author表:

MySQL查詢語句中的IN 和Exists 對比分析

 

使用exists時,數(shù)據(jù)量的變化沒有帶來執(zhí)行計劃的改變,但由于子查詢結果集很大,5.5以后的MySQL版本在exists匹配查詢結果時使用的是Block Nested-Loop(Block嵌套循環(huán),引入join buffer,類似于緩存功能)開始對查詢效率產(chǎn)生顯著影響,尤其針對<font color=red>子查詢結果集很大</font>的情況下能顯著改善查詢匹配效率:

MySQL查詢語句中的IN 和Exists 對比分析

 

實驗結論

根據(jù)上述兩個實驗及實驗結果,我們可以較清晰的理解IN 和Exists的執(zhí)行過程,并歸納出IN 和Exists的適用場景:

  • IN查詢在內部表和外部表上都可以使用到索引;
  • Exists查詢僅在內部表上可以使用到索引;
  • 子查詢結果集很大,而外部表較小的時候,Exists的Block Nested Loop(Block 嵌套循環(huán))的作用開始顯現(xiàn),并彌補外部表無法用到索引的缺陷,查詢效率會優(yōu)于IN。
  • 子查詢結果集較小,而外部表很大的時候,Exists的Block嵌套循環(huán)優(yōu)化效果不明顯,IN 的外表索引優(yōu)勢占主要作用,此時IN的查詢效率會優(yōu)于Exists。
  • 網(wǎng)上的說法不準確。其實“表的規(guī)模”不是看內部表和外部表,而是外部表和子查詢結果集。
  • 最后一點,也是最重要的一點:世間沒有絕對的真理,掌握事物的本質,針對不同的場景進行實踐驗證才是最可靠有效的方法。

實驗過程中發(fā)現(xiàn)的問題補充

僅對不同數(shù)據(jù)集情況下的上述exists語句分析時發(fā)現(xiàn),數(shù)據(jù)集越大,消耗的時間反而變小,覺得很奇怪。

具體查詢條件為:

where tp.poetry_id>3650,耗時0.13S

where tp.poetry_id>293650,耗時0.46S

可能原因:條件值大,查詢越靠后,需要遍歷的記錄越多,造成最終消耗越多的時間。這個解釋有待進一步驗證后再補充。 

責任編輯:龐桂玉 來源: 今日頭條
相關推薦

2010-11-25 15:57:49

mysql查詢語句

2010-07-14 10:26:58

IMAP協(xié)議

2018-01-26 14:29:01

框架

2018-01-21 14:11:22

人工智能PaddlePaddlTensorflow

2023-05-14 22:00:01

2010-06-08 11:15:43

OpenSUSE Ub

2010-08-04 15:47:24

NFS版本

2016-10-18 21:10:17

GitHubBitbucketGitLab

2010-07-20 16:16:21

SDH

2024-08-08 07:38:42

2015-03-09 15:06:20

javaphpweb開發(fā)

2017-03-20 14:32:57

2017-05-05 10:15:38

深度學習框架對比分析

2010-06-24 21:35:33

2013-01-17 16:11:11

數(shù)據(jù)中心交換機網(wǎng)絡虛擬化

2021-05-18 10:18:15

Java

2021-03-15 08:40:46

數(shù)據(jù)分析波動

2023-06-05 07:35:03

2023-05-18 07:30:16

OpenCLGPU平臺生態(tài)

2011-09-08 16:30:59

SQL Server查詢
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 中文字幕亚洲区一区二 | 国产精品精品视频一区二区三区 | 国产高清自拍视频在线观看 | 久久久久国产视频 | 色综合天天天天做夜夜夜夜做 | 国产精品免费一区二区三区 | 在线免费观看黄网 | 请别相信他免费喜剧电影在线观看 | 午夜国产在线 | 精品国产欧美 | 久久69精品久久久久久久电影好 | 国产一区二区不卡 | 国产精品一区二区在线 | 亚洲一区中文字幕在线观看 | 午夜视频一区 | 丁香久久 | 国产日韩欧美91 | 欧美自拍视频 | 中文字幕免费观看 | 中文字幕一区二区在线观看 | 国产一区二区三区高清 | 亚洲第一免费播放区 | 国产日韩视频 | 亚洲a级 | 国产精品精品视频一区二区三区 | 久久精品免费 | 色在线看 | 综合久| 精品久久久久久亚洲精品 | 久久国产精品-久久精品 | 九九激情视频 | 日韩欧美一区二区三区免费观看 | 免费在线观看91 | 在线观看成人 | 欧美日韩国产精品一区 | 性一区 | 日韩一区二区三区在线观看视频 | 天天插天天操 | 亚洲国产二区 | 精品粉嫩aⅴ一区二区三区四区 | 精品福利在线 |