《Mysql應(yīng)用MySQL的LEFT JOIN表連接的進(jìn)階學(xué)習(xí)教程》要點(diǎn):
本文介紹了Mysql應(yīng)用MySQL的LEFT JOIN表連接的進(jìn)階學(xué)習(xí)教程,希望對(duì)您有用。如果有疑問,可以聯(lián)系我們。
LEFT JOIN的主表MYSQL數(shù)據(jù)庫
這里所說的主表是指在連接查詢里MySQL以哪個(gè)表為主進(jìn)行查詢.比如說在LEFT JOIN查詢里,一般來說左表就是主表,但這只是經(jīng)驗(yàn)之談,很多時(shí)候經(jīng)驗(yàn)主義是靠不住的,為了說明問題,先來個(gè)例子,建兩個(gè)演示用的表categories和posts:MYSQL數(shù)據(jù)庫
CREATE TABLE IF NOT EXISTS `categories` ( `id` int(10) unsigned NOT NULL AUTO_INCREMENT, `name` varchar(15) NOT NULL, `created` datetime NOT NULL, PRIMARY KEY (`id`) ); CREATE TABLE IF NOT EXISTS `posts` ( `id` int(10) unsigned NOT NULL AUTO_INCREMENT, `category_id` int(10) unsigned NOT NULL, `title` varchar(100) NOT NULL, `content` varchar(200) NOT NULL, `created` datetime NOT NULL, PRIMARY KEY (`id`), KEY `category_id` (`category_id`) );
先注意一下每個(gè)表的索引情況,以后會(huì)用到,記得隨便插入一點(diǎn)測試數(shù)據(jù),不用太多,但怎么也得兩行以上,然后執(zhí)行以下SQL:MYSQL數(shù)據(jù)庫
EXPLAIN SELECT * FROM posts LEFT JOIN categories ON posts.category_id = categories.id WHERE categories.id = ‘一個(gè)已經(jīng)存在的ID' ORDER BY posts.created DESC
table key Extra categories PRIMARY Using filesort posts category_id Using where
在explain的結(jié)果中,第一行表示的表就是主表,所以說在此查詢里categories是主表,而在我們的經(jīng)驗(yàn)里,LEFT JOIN查詢里,左表(posts表)才應(yīng)該是主表,這產(chǎn)生一個(gè)根本的矛盾,MySQL之所以這樣處理,是因?yàn)樵谖覀兊腤HERE部分,查詢條件是按照categories表的字段來進(jìn)行篩選的,而恰恰categories表存在合適的索引,所以在查詢時(shí)把categories表作為主表更有利于縮小結(jié)果集.MYSQL數(shù)據(jù)庫
那explain結(jié)果中的Using filesort又是為什么呢?這是因?yàn)橹鞅硎莄ategories表,從表是posts表,而我們使用從表的字段去ORDER BY,這通常不是一個(gè)好選擇,最好改成主表字段,如果鑒于需求所限,無法改成主表的字段,那么可以嘗試添加如下索引:MYSQL數(shù)據(jù)庫
ALTER TABLE `posts` ADD INDEX ( `category_id` , `created` );
再運(yùn)行SQL時(shí)就不會(huì)有Using filesort了,這是因?yàn)橹鞅韈ategories在通過category_id連接從表posts時(shí),可以進(jìn)而通過索引直接得到排序后的posts結(jié)果.MYSQL數(shù)據(jù)庫
主觀上一旦搞錯(cuò)了主表,可能怎么調(diào)整索引都得不到高效的SQL,所以在寫SQL時(shí),比如說在寫LEFT JOIN查詢時(shí),如果希望左表是主表,那么就要保證在WHERE語句里的查詢條件盡可能多的使用左表字段,進(jìn)而,一旦確定了主表,也最好只通過主表字段去ORDER BY.MYSQL數(shù)據(jù)庫
LEFT JOIN查詢效率分析
user表:MYSQL數(shù)據(jù)庫
id | name --------- 1 | libk 2 | zyfon 3 | daodao user_action表: user_id | action --------------- 1 | jump 1 | kick 1 | jump 2 | run 4 | swim
sql:
MYSQL數(shù)據(jù)庫
select id, name, action from user as u left join user_action a on u.id = a.user_id result: id | name | action -------------------------------- 1 | libk | jump ① 1 | libk | kick ② 1 | libk | jump ③ 2 | zyfon | run ④ 3 | daodao | null ⑤
分析:
注意到user_action中還有一個(gè)user_id=4, action=swim的紀(jì)錄,但是沒有在結(jié)果中出現(xiàn),
而user表中的id=3, name=daodao的用戶在user_action中沒有相應(yīng)的紀(jì)錄,但是卻出現(xiàn)在了結(jié)果集中
因?yàn)楝F(xiàn)在是left join,所有的工作以left為準(zhǔn).
結(jié)果1,2,3,4都是既在左表又在右表的紀(jì)錄,5是只在左表,不在右表的紀(jì)錄MYSQL數(shù)據(jù)庫
結(jié)論:
我們可以想象left join 是這樣工作的
從左表讀出一條,選出所有與on匹配的右表紀(jì)錄(n條)進(jìn)行連接,形成n條紀(jì)錄(包括重復(fù)的行,如:結(jié)果1和結(jié)果3),
如果右邊沒有與on條件匹配的表,那連接的字段都是null.
然后繼續(xù)讀下一條.MYSQL數(shù)據(jù)庫
引申:
我們可以用右表沒有on匹配則顯示null的規(guī)律, 來找出所有在左表,不在右表的紀(jì)錄, 注意用來判斷的那列必須聲明為not null的.
如:
MYSQL數(shù)據(jù)庫
select id, name, action from user as u left join user_action a on u.id = a.user_id where a.user_id is NULL
(注意:1.列值為null應(yīng)該用is null 而不能用=NULL
2.這里a.user_id 列必須聲明為 NOT NULL 的)
MYSQL數(shù)據(jù)庫
result: id | name | action -------------------------- 3 | daodao | NULL --------------------------------------------------------------------------------
Tips:
1. on a.c1 = b.c1 等同于 using(c1)
2. INNER JOIN 和 , (逗號(hào)) 在語義上是等同的
3. 當(dāng) MySQL 在從一個(gè)表中檢索信息時(shí),你可以提示它選擇了哪一個(gè)索引.
如果 EXPLAIN 顯示 MySQL 使用了可能的索引列表中錯(cuò)誤的索引,這個(gè)特性將是很有用的.
通過指定 USE INDEX (key_list),你可以告訴 MySQL 使用可能的索引中最合適的一個(gè)索引在表中查找記錄行.
可選的二選一句法 IGNORE INDEX (key_list) 可被用于告訴 MySQL 不使用特定的索引.
4. 一些例子:
MYSQL數(shù)據(jù)庫
mysql> SELECT * FROM table1,table2 WHERE table1.id=table2.id; mysql> SELECT * FROM table1 LEFT JOIN table2 ON table1.id=table2.id; mysql> SELECT * FROM table1 LEFT JOIN table2 USING (id); mysql> SELECT * FROM table1 LEFT JOIN table2 ON table1.id=table2.id -> LEFT JOIN table3 ON table2.id=table3.id; mysql> SELECT * FROM table1 USE INDEX (key1,key2) -> WHERE key1=1 AND key2=2 AND key3=3; mysql> SELECT * FROM table1 IGNORE INDEX (key3) -> WHERE key1=1 AND key2=2 AND key3=3;
轉(zhuǎn)載請(qǐng)注明本頁網(wǎng)址:
http://www.fzlkiss.com/jiaocheng/5010.html