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

對(duì)比分析MySQL語(yǔ)句中的IN 和Exists

 更新時(shí)間:2018年06月05日 08:50:24   作者:lilugoodjob  
mysql中in 是把外表和內(nèi)表作hash 連接,而exists是對(duì)外表作loop循環(huán),每次loop循環(huán)再對(duì)內(nèi)表進(jìn)行查詢(xún)。一直以來(lái)認(rèn)為exists比in效率高的說(shuō)法是不準(zhǔn)確的。

背景介紹

最近在寫(xiě)SQL語(yǔ)句時(shí),對(duì)選擇IN 還是Exists 猶豫不決,于是把兩種方法的SQL都寫(xiě)出來(lái)對(duì)比一下執(zhí)行效率,發(fā)現(xiàn)IN的查詢(xún)效率比Exists高了很多,于是想當(dāng)然的認(rèn)為IN的效率比Exists好,但本著尋根究底的原則,我想知道這個(gè)結(jié)論是否適用所有場(chǎng)景,以及為什么會(huì)出現(xiàn)這個(gè)結(jié)果。
網(wǎng)上查了一下相關(guān)資料,大體可以歸納為:外部表小,內(nèi)部表大時(shí),適用Exists;外部表大,內(nèi)部表小時(shí),適用IN。那我就困惑了,因?yàn)槲业腟QL語(yǔ)句里面,外表只有1W級(jí)別的數(shù)據(jù),內(nèi)表有30W級(jí)別的數(shù)據(jù),按網(wǎng)上的說(shuō)法應(yīng)該是Exists的效率會(huì)比IN高的,但我的結(jié)果剛好相反?。?br /> “沒(méi)有調(diào)查就沒(méi)有發(fā)言權(quán)”!于是我開(kāi)始研究IN 和Exists的實(shí)際執(zhí)行過(guò)程,從實(shí)踐的角度出發(fā),在根本上去尋找原因,于是有了這篇博文分享。

實(shí)驗(yàn)數(shù)據(jù)

我的實(shí)驗(yàn)數(shù)據(jù)包括兩張表:t_author表 和 t_poetry表。
對(duì)應(yīng)表的數(shù)據(jù)量:

t_author表,13355條記錄;
t_poetry表,289917條記錄。

對(duì)應(yīng)的表結(jié)構(gòu)如下:

CREATE TABLE t_poetry (
id bigint(20) NOT NULL AUTO_INCREMENT,
poetry_id bigint(20) NOT NULL COMMENT '詩(shī)詞id',
poetry_name varchar(200) NOT NULL COMMENT '詩(shī)詞名稱(chēng)',
<font color=red> author_id bigint(20) NOT NULL COMMENT '作者id'</font>
PRIMARY KEY (id),
UNIQUE KEY pid_idx (poetry_id) USING BTREE,
KEY aid_idx (author_id) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=291270 DEFAULT CHARSET=utf8mb4

CREATE TABLE t_author (
id int(15) NOT NULL AUTO_INCREMENT,
author_id bigint(20) NOT NULL,</font>
author_name varchar(32) NOT NULL,
dynasty varchar(16) NOT NULL,
poetry_num int(8) NOT NULL DEFAULT '0'
PRIMARY KEY (id),
<font color=red>UNIQUE KEY authorid_idx (author_id) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=13339 DEFAULT CHARSET=utf8mb4

執(zhí)行計(jì)劃分析 IN 執(zhí)行過(guò)程

sql示例:select * from tabA where tabA.x in (select x from tabB where y>0 );

其執(zhí)行計(jì)劃:
(1)執(zhí)行tabB表的子查詢(xún),得到結(jié)果集B,可以使用到tabB表的索引y;
(2)執(zhí)行tabA表的查詢(xún),查詢(xún)條件是tabA.x在結(jié)果集B里面,可以使用到tabA表的索引x。

Exists執(zhí)行過(guò)程

sql示例:select from tabA where exists (select from tabB where y>0);

其執(zhí)行計(jì)劃:

(1)先將tabA表所有記錄取到。
(2)逐行針對(duì)tabA表的記錄,去關(guān)聯(lián)tabB表,判斷tabB表的子查詢(xún)是否有返回?cái)?shù)據(jù),5.5之后的版本使用Block Nested Loop(Block 嵌套循環(huán))。
(3)如果子查詢(xún)有返回?cái)?shù)據(jù),則將tabA當(dāng)前記錄返回到結(jié)果集。
tabA相當(dāng)于取全表數(shù)據(jù)遍歷,tabB可以使用到索引。

實(shí)驗(yàn)過(guò)程

實(shí)驗(yàn)針對(duì)相同結(jié)果集的IN和Exists 的SQL語(yǔ)句進(jìn)行分析。
包含IN的SQL語(yǔ)句:

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

包含Exists的SQL語(yǔ)句:

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

第一次實(shí)驗(yàn)數(shù)據(jù)情況

t_author表,13355條記錄;t_poetry表,子查詢(xún)篩選結(jié)果集 where poetry_id>293650 ,121條記錄;

執(zhí)行結(jié)果

使用exists耗時(shí)0.94S, 使用in耗時(shí)0.03S,IN 效率高于Exists

原因分析

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


使用exists時(shí),t_author表全表掃描:


在子查詢(xún)結(jié)果集較小時(shí),查詢(xún)耗時(shí)主要表現(xiàn)在對(duì)t_author表的遍歷上。

第二次實(shí)驗(yàn)數(shù)據(jù)情況

t_author表,13355條記錄;t_poetry表,子查詢(xún)篩選結(jié)果集 where poetry_id>3650 ,287838條記錄;

執(zhí)行時(shí)間

使用exists耗時(shí)0.12S, 使用in耗時(shí)0.48S,Exists效率高于 IN

原因分析

兩者的索引使用情況跟第一次實(shí)驗(yàn)是一致的,唯一區(qū)別是子查詢(xún)篩選結(jié)果集的大小不同,但實(shí)驗(yàn)結(jié)果已經(jīng)跟第一次的不同了。這種情況下子查詢(xún)結(jié)果集很大,我們看看mysql的查詢(xún)計(jì)劃:
使用in時(shí),由于子查詢(xún)結(jié)果集很大,對(duì)t_author和t_poetry表都接近于全表掃描,此時(shí)對(duì)t_author表的遍歷耗時(shí)差異對(duì)整體效率影響可以忽略,執(zhí)行計(jì)劃里多了一行<auto_key>,在接近全表掃描的情況下,mysql優(yōu)化器選擇了auto_key來(lái)遍歷t_author表:

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

實(shí)驗(yàn)結(jié)論

根據(jù)上述兩個(gè)實(shí)驗(yàn)及實(shí)驗(yàn)結(jié)果,我們可以較清晰的理解IN 和Exists的執(zhí)行過(guò)程,并歸納出IN 和Exists的適用場(chǎng)景:

IN查詢(xún)?cè)趦?nèi)部表和外部表上都可以使用到索引; Exists查詢(xún)僅在內(nèi)部表上可以使用到索引;當(dāng)子查詢(xún)結(jié)果集很大,而外部表較小的時(shí)候,Exists的Block Nested Loop(Block 嵌套循環(huán))的作用開(kāi)始顯現(xiàn),并彌補(bǔ)外部表無(wú)法用到索引的缺陷,查詢(xún)效率會(huì)優(yōu)于IN。當(dāng)子查詢(xún)結(jié)果集較小,而外部表很大的時(shí)候,Exists的Block嵌套循環(huán)優(yōu)化效果不明顯,IN 的外表索引優(yōu)勢(shì)占主要作用,此時(shí)IN的查詢(xún)效率會(huì)優(yōu)于Exists。 網(wǎng)上的說(shuō)法不準(zhǔn)確。其實(shí)“表的規(guī)?!辈皇强磧?nèi)部表和外部表,而是外部表和子查詢(xún)結(jié)果集。最后一點(diǎn),也是最重要的一點(diǎn):世間沒(méi)有絕對(duì)的真理,掌握事物的本質(zhì),針對(duì)不同的場(chǎng)景進(jìn)行實(shí)踐驗(yàn)證才是最可靠有效的方法。 實(shí)驗(yàn)過(guò)程中發(fā)現(xiàn)的問(wèn)題補(bǔ)充

僅對(duì)不同數(shù)據(jù)集情況下的上述exists語(yǔ)句分析時(shí)發(fā)現(xiàn),數(shù)據(jù)集越大,消耗的時(shí)間反而變小,覺(jué)得很奇怪。
具體查詢(xún)條件為:

where tp.poetry_id>3650,耗時(shí)0.13S
where tp.poetry_id>293650,耗時(shí)0.46S

可能原因:條件值大,查詢(xún)?cè)娇亢?,需要遍歷的記錄越多,造成最終消耗越多的時(shí)間。這個(gè)解釋有待進(jìn)一步驗(yàn)證后再補(bǔ)充。

相關(guān)文章

  • MySQL中預(yù)處理語(yǔ)句prepare、execute與deallocate的使用教程

    MySQL中預(yù)處理語(yǔ)句prepare、execute與deallocate的使用教程

    這篇文章主要介紹了MySQL中預(yù)處理語(yǔ)句prepare、execute與deallocate的使用教程,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用mysql具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面跟著小編一起來(lái)學(xué)習(xí)學(xué)習(xí)吧。
    2017-08-08
  • mysql并發(fā)控制原理知識(shí)點(diǎn)

    mysql并發(fā)控制原理知識(shí)點(diǎn)

    在本篇文章里小編給大家整理的是一篇關(guān)于mysql并發(fā)控制原理知識(shí)點(diǎn)內(nèi)容,需要的朋友們參考下吧。
    2020-02-02
  • Django2.* + Mysql5.7開(kāi)發(fā)環(huán)境整合教程圖解

    Django2.* + Mysql5.7開(kāi)發(fā)環(huán)境整合教程圖解

    這篇文章主要介紹了Django2.* + Mysql5.7開(kāi)發(fā)環(huán)境整合教程,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2019-09-09
  • 解決MySQL報(bào)錯(cuò):You?can‘t?specify?target?table?‘region‘?for?update?in?FROM?clause

    解決MySQL報(bào)錯(cuò):You?can‘t?specify?target?table?‘region‘?for?

    這篇文章主要給大家介紹了關(guān)于MySQL報(bào)錯(cuò):You?can‘t?specify?target?table?‘region‘?for?update?in?FROM?clause的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-02-02
  • DBeaver如何實(shí)現(xiàn)導(dǎo)入excel中的大量數(shù)據(jù)

    DBeaver如何實(shí)現(xiàn)導(dǎo)入excel中的大量數(shù)據(jù)

    使用DBeaver導(dǎo)入Excel數(shù)據(jù)需先將文件轉(zhuǎn)換為CSV格式,詳細(xì)步驟包括:將Excel文件另存為CSV,確保列名與數(shù)據(jù)庫(kù)表字段對(duì)應(yīng),然后在DBeaver中創(chuàng)建表和導(dǎo)入CSV文件,注意選擇正確的編碼格式以防中文亂碼
    2024-10-10
  • mysql 教程 存儲(chǔ)過(guò)程

    mysql 教程 存儲(chǔ)過(guò)程

    最近用mysql + asp.net來(lái)寫(xiě)網(wǎng)站,既然mysql已經(jīng)支持存儲(chǔ)過(guò)程了,那么像分頁(yè)這么常用的東西,當(dāng)然要用存儲(chǔ)過(guò)程啦
    2009-06-06
  • MySQL中的批量修改、插入操作數(shù)據(jù)庫(kù)

    MySQL中的批量修改、插入操作數(shù)據(jù)庫(kù)

    在平常的項(xiàng)目中,我們會(huì)需要批量操作數(shù)據(jù)庫(kù)的時(shí)候,例如:批量修改,批量插入,那我們不應(yīng)該使用 for 循環(huán)去操作數(shù)據(jù)庫(kù),這樣會(huì)導(dǎo)致我們反復(fù)與數(shù)據(jù)庫(kù)發(fā)生連接和斷開(kāi)連接,影響性能和增加操作時(shí)間,所以可以使用SQL 批量修改的方式去操作數(shù)據(jù)庫(kù),感興趣的朋友一起學(xué)習(xí)下吧
    2023-09-09
  • MySQL5.7.20解壓版安裝和修改root密碼的教程

    MySQL5.7.20解壓版安裝和修改root密碼的教程

    這篇文章主要介紹了MySQL5.7.20解壓版安裝和修改root密碼的教程,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下
    2018-04-04
  • mysql高效導(dǎo)數(shù)據(jù)的方法講解

    mysql高效導(dǎo)數(shù)據(jù)的方法講解

    模擬現(xiàn)網(wǎng)測(cè)試,需要搭建測(cè)試環(huán)境,導(dǎo)入上億級(jí)的數(shù)據(jù)到數(shù)據(jù)庫(kù)。對(duì)于到的問(wèn)題做些簡(jiǎn)單記錄,有需要的朋友可以參考一下
    2013-09-09
  • update.where無(wú)索引導(dǎo)致MySQL死鎖問(wèn)題解決

    update.where無(wú)索引導(dǎo)致MySQL死鎖問(wèn)題解決

    這篇文章主要為大家介紹了update.where無(wú)索引導(dǎo)致MySQL死鎖問(wèn)題解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-11-11

最新評(píng)論