淺析MySQL中exists與in的使用

exists對外表用loop逐條查詢梯浪,每次查詢都會(huì)查看exists的條件語句,當(dāng) exists里的條件語句能夠返回記錄行時(shí)(無論記錄行是的多少礼预,只要能返回)虏劲,條件就為真,返回當(dāng)前l(fā)oop到的這條記錄励堡,反之如果exists里的條 件語句不能返回記錄行吻育,則當(dāng)前l(fā)oop到的這條記錄被丟棄,exists的條件就像一個(gè)bool條件摊趾,當(dāng)能返回結(jié)果集則為true,不能返回結(jié)果集則為 false

如下:

select * from user where exists (select 1);

對user表的記錄逐條取出漩绵,由于子條件中的select 1永遠(yuǎn)能返回記錄行肛炮,那么user表的所有記錄都將被加入結(jié)果集,所以與 select * from user;是一樣的

又如下

select * from user where exists (select * from user where userId = 0);

可以知道對user表進(jìn)行l(wèi)oop時(shí)碍扔,檢查條件語句(select * from user where userId = 0),由于userId永遠(yuǎn)不為0秕重,所以條件語句永遠(yuǎn)返回空集,條件永遠(yuǎn)為false二拐,那么user表的所有記錄都將被丟棄

not exists與exists相反凳兵,也就是當(dāng)exists條件有結(jié)果集返回時(shí)庐扫,loop到的記錄將被丟棄,否則將loop到的記錄加入結(jié)果集

總的來說聚蝶,如果A表有n條記錄,那么exists查詢就是將這n條記錄逐條取出,然后判斷n遍exists條件?



in查詢相當(dāng)于多個(gè)or條件的疊加验靡,這個(gè)比較好理解雏节,比如下面的查詢

select * from user where userId in (1, 2, 3);

等效于

select * from user where userId = 1 or userId = 2 or userId = 3;

not in與in相反,如下

select * from user where userId not in (1, 2, 3);

等效于

select * from user where userId != 1 and userId != 2 and userId != 3;

總的來說辞州,in查詢就是先將子查詢條件的記錄全都查出來寥粹,假設(shè)結(jié)果集為B,共有m條記錄媚狰,然后在將子查詢條件的結(jié)果集分解成m個(gè),再進(jìn)行m次查詢


值得一提的是类嗤,in查詢的子條件返回結(jié)果必須只有一個(gè)字段辨宠,例如

select * from user where userId in (select id from B);

而不能是

select * from user where userId in (select id, age from B);

而exists就沒有這個(gè)限制


下面來考慮exists和in的性能

考慮如下SQL語句

1: select * from A where exists (select * from B where B.id = A.id);

2: select * from A where A.id in (select id from B);


查詢1.可以轉(zhuǎn)化以下偽代碼嗤形,便于理解

for ($i = 0; $i < count(A); $i++) {

  $a = get_record(A, $i); #從A表逐條獲取記錄

  if (B.id = $a[id]) #如果子條件成立

    $result[] = $a;

}

return $result;

大概就是這么個(gè)意思黄伊,其實(shí)可以看到,查詢1主要是用到了B表的索引,A表如何對查詢的效率影響應(yīng)該不大


假設(shè)B表的所有id為1,2,3,查詢2可以轉(zhuǎn)換為

select * from A where A.id = 1 or A.id = 2 or A.id = 3;

這個(gè)好理解了派殷,這里主要是用到了A的索引还最,B表如何對查詢影響不大


下面再看not exists 和 not in

1. select * from A where not exists (select * from B where B.id = A.id);

2. select * from A where A.id not in (select id from B);

看查詢1,還是和上面一樣毡惜,用了B的索引

而對于查詢2拓轻,可以轉(zhuǎn)化成如下語句

select * from A where A.id != 1 and A.id != 2 and A.id != 3;

可以知道not in是個(gè)范圍查詢,這種!=的范圍查詢無法使用任何索引,等于說A表的每條記錄经伙,都要在B表里遍歷一次扶叉,查看B表里是否存在這條記錄

故not exists比not in效率高


mysql中的in語句是把外表和內(nèi)表作hash 連接,而exists語句是對外表作loop循環(huán)帕膜,每次loop循環(huán)再對內(nèi)表進(jìn)行查詢枣氧。一直大家都認(rèn)為exists比in語句的效率要高,這種說法其實(shí)是不準(zhǔn)確的达吞。這個(gè)是要區(qū)分環(huán)境的。

如果查詢的兩個(gè)表大小相當(dāng)荒典,那么用in和exists差別不大酪劫。

如果兩個(gè)表中一個(gè)較小,一個(gè)是大表寺董,則子查詢表大的用exists覆糟,子查詢表小的用in:?

例如:表A(小表),表B(大表)


1:

select * from A where cc in (select cc from B)效率低遮咖,用到了A表上cc列的索引滩字;


select * from A where exists(select cc from B where cc=A.cc)效率高,用到了B表上cc列的索引。

相反的


2:

select * from B where cc in (select cc from A)效率高麦箍,用到了B表上cc列的索引漓藕;


select * from B where exists(select cc from A where cc=B.cc)?效率低,用到了A表上cc列的索引内列。



not in 和not exists如果查詢語句使用了not in 那么內(nèi)外表都進(jìn)行全表掃描撵术,沒有用到索引;而not extsts 的子查詢依然能用到表上的索引话瞧。所以無論那個(gè)表大嫩与,用not exists都比not in要快

in 與 =的區(qū)別?

select name from student where name in ('zhang','wang','li','zhao');?

與?

select name from student where name='zhang' or name='li' or name='wang' or name='zhao'?

的結(jié)果是相同的交排。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末划滋,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子埃篓,更是在濱河造成了極大的恐慌处坪,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,919評(píng)論 6 502
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件架专,死亡現(xiàn)場離奇詭異同窘,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)部脚,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,567評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門想邦,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人委刘,你說我怎么就攤上這事丧没。” “怎么了锡移?”我有些...
    開封第一講書人閱讀 163,316評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵呕童,是天一觀的道長。 經(jīng)常有香客問我淆珊,道長夺饲,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,294評(píng)論 1 292
  • 正文 為了忘掉前任套蒂,我火速辦了婚禮钞支,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘操刀。我一直安慰自己,他們只是感情好婴洼,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,318評(píng)論 6 390
  • 文/花漫 我一把揭開白布骨坑。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪欢唾。 梳的紋絲不亂的頭發(fā)上且警,一...
    開封第一講書人閱讀 51,245評(píng)論 1 299
  • 那天,我揣著相機(jī)與錄音礁遣,去河邊找鬼斑芜。 笑死,一個(gè)胖子當(dāng)著我的面吹牛祟霍,可吹牛的內(nèi)容都是我干的杏头。 我是一名探鬼主播,決...
    沈念sama閱讀 40,120評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼沸呐,長吁一口氣:“原來是場噩夢啊……” “哼醇王!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起崭添,我...
    開封第一講書人閱讀 38,964評(píng)論 0 275
  • 序言:老撾萬榮一對情侶失蹤寓娩,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后呼渣,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體棘伴,經(jīng)...
    沈念sama閱讀 45,376評(píng)論 1 313
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,592評(píng)論 2 333
  • 正文 我和宋清朗相戀三年屁置,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了焊夸。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片赘理。...
    茶點(diǎn)故事閱讀 39,764評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡晃酒,死狀恐怖橄教,靈堂內(nèi)的尸體忽然破棺而出枪汪,到底是詐尸還是另有隱情吁恍,我是刑警寧澤装畅,帶...
    沈念sama閱讀 35,460評(píng)論 5 344
  • 正文 年R本政府宣布儡遮,位于F島的核電站你虹,受9級(jí)特大地震影響并徘,放射性物質(zhì)發(fā)生泄漏遣钳。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,070評(píng)論 3 327
  • 文/蒙蒙 一麦乞、第九天 我趴在偏房一處隱蔽的房頂上張望蕴茴。 院中可真熱鬧,春花似錦姐直、人聲如沸倦淀。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,697評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽撞叽。三九已至姻成,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間愿棋,已是汗流浹背科展。 一陣腳步聲響...
    開封第一講書人閱讀 32,846評(píng)論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留糠雨,地道東北人才睹。 一個(gè)月前我還...
    沈念sama閱讀 47,819評(píng)論 2 370
  • 正文 我出身青樓,卻偏偏與公主長得像甘邀,于是被迫代替她去往敵國和親琅攘。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,665評(píng)論 2 354