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

MySQL的LEFT JOIN表連接的進(jìn)階學(xué)習(xí)教程

 更新時(shí)間:2015年12月09日 10:38:10   作者:KimiChen  
這篇文章主要介紹了MySQL的LEFT JOIN表連接的進(jìn)階學(xué)習(xí)教程,包括對(duì)左連接的查詢效率分析以及相關(guān)建議,需要的朋友可以參考下

LEFT JOIN的主表

這里所說(shuō)的主表是指在連接查詢里MySQL以哪個(gè)表為主進(jìn)行查詢。比如說(shuō)在LEFT JOIN查詢里,一般來(lái)說(shuō)左表就是主表,但這只是經(jīng)驗(yàn)之談,很多時(shí)候經(jīng)驗(yàn)主義是靠不住的,為了說(shuō)明問(wèn)題,先來(lái)個(gè)例子,建兩個(gè)演示用的表categories和posts:

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)測(cè)試數(shù)據(jù),不用太多,但怎么也得兩行以上,然后執(zhí)行以下SQL:

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é)果中,第一行表示的表就是主表,所以說(shuō)在此查詢里categories是主表,而在我們的經(jīng)驗(yàn)里,LEFT JOIN查詢里,左表(posts表)才應(yīng)該是主表,這產(chǎn)生一個(gè)根本的矛盾,MySQL之所以這樣處理,是因?yàn)樵谖覀兊腤HERE部分,查詢條件是按照categories表的字段來(lái)進(jìn)行篩選的,而恰恰categories表存在合適的索引,所以在查詢時(shí)把categories表作為主表更有利于縮小結(jié)果集。

那explain結(jié)果中的Using filesort又是為什么呢?這是因?yàn)橹鞅硎莄ategories表,從表是posts表,而我們使用從表的字段去ORDER BY,這通常不是一個(gè)好選擇,最好改成主表字段,如果鑒于需求所限,無(wú)法改成主表的字段,那么可以嘗試添加如下索引:

ALTER TABLE `posts` ADD INDEX ( `category_id` , `created` );

再運(yùn)行SQL時(shí)就不會(huì)有Using filesort了,這是因?yàn)橹鞅韈ategories在通過(guò)category_id連接從表posts時(shí),可以進(jìn)而通過(guò)索引直接得到排序后的posts結(jié)果。

主觀上一旦搞錯(cuò)了主表,可能怎么調(diào)整索引都得不到高效的SQL,所以在寫SQL時(shí),比如說(shuō)在寫LEFT JOIN查詢時(shí),如果希望左表是主表,那么就要保證在WHERE語(yǔ)句里的查詢條件盡可能多的使用左表字段,進(jìn)而,一旦確定了主表,也最好只通過(guò)主表字段去ORDER BY。

LEFT JOIN查詢效率分析
user表:

id | name
---------
1 | libk
2 | zyfon
3 | daodao

user_action表:

user_id | action
---------------
1 | jump
1 | kick
1 | jump
2 | run
4 | swim

sql:

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ì)錄,但是沒(méi)有在結(jié)果中出現(xiàn),
而user表中的id=3, name=daodao的用戶在user_action中沒(méi)有相應(yīng)的紀(jì)錄,但是卻出現(xiàn)在了結(jié)果集中
因?yàn)楝F(xiàn)在是left join,所有的工作以left為準(zhǔn).
結(jié)果1,2,3,4都是既在左表又在右表的紀(jì)錄,5是只在左表,不在右表的紀(jì)錄


結(jié)論:
我們可以想象left join 是這樣工作的
從左表讀出一條,選出所有與on匹配的右表紀(jì)錄(n條)進(jìn)行連接,形成n條紀(jì)錄(包括重復(fù)的行,如:結(jié)果1和結(jié)果3),
如果右邊沒(méi)有與on條件匹配的表,那連接的字段都是null.
然后繼續(xù)讀下一條。

引申:
我們可以用右表沒(méi)有on匹配則顯示null的規(guī)律, 來(lái)找出所有在左表,不在右表的紀(jì)錄, 注意用來(lái)判斷的那列必須聲明為not null的。
如:

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 的)

result:
id | name | action
--------------------------
3 | daodao | NULL

--------------------------------------------------------------------------------

Tips:
1. on a.c1 = b.c1 等同于 using(c1)
2. INNER JOIN 和 , (逗號(hào)) 在語(yǔ)義上是等同的
3. 當(dāng) MySQL 在從一個(gè)表中檢索信息時(shí),你可以提示它選擇了哪一個(gè)索引。
如果 EXPLAIN 顯示 MySQL 使用了可能的索引列表中錯(cuò)誤的索引,這個(gè)特性將是很有用的。
通過(guò)指定 USE INDEX (key_list),你可以告訴 MySQL 使用可能的索引中最合適的一個(gè)索引在表中查找記錄行。
可選的二選一句法 IGNORE INDEX (key_list) 可被用于告訴 MySQL 不使用特定的索引。
4. 一些例子:

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;

相關(guān)文章

  • Mysql查詢數(shù)據(jù)庫(kù)或數(shù)據(jù)表中的數(shù)據(jù)量以及數(shù)據(jù)大小

    Mysql查詢數(shù)據(jù)庫(kù)或數(shù)據(jù)表中的數(shù)據(jù)量以及數(shù)據(jù)大小

    許多數(shù)據(jù)庫(kù)的元數(shù)據(jù)都是存儲(chǔ)在mysql中的,本文主要介紹了Mysql查詢數(shù)據(jù)庫(kù)或數(shù)據(jù)表中的數(shù)據(jù)量以及數(shù)據(jù)大小,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2024-02-02
  • MySQL高效可靠處理持久化數(shù)據(jù)的教程指南

    MySQL高效可靠處理持久化數(shù)據(jù)的教程指南

    這篇文章主要給大家詳細(xì)介紹了 MySQL 如何高效可靠處理持久化數(shù)據(jù),文中有詳細(xì)的流程步驟和代碼示例,對(duì)我們的學(xué)習(xí)有一定的幫助,需要的朋友可以參考下
    2023-07-07
  • mysql計(jì)算時(shí)間差函數(shù)

    mysql計(jì)算時(shí)間差函數(shù)

    MySql計(jì)算兩個(gè)日期的時(shí)間差函數(shù)TIMESTAMPDIFF用法,只要用一句SQL語(yǔ)句就可以辦到了。
    2013-11-11
  • mysql中文顯示為問(wèn)號(hào)?的問(wèn)題及解決

    mysql中文顯示為問(wèn)號(hào)?的問(wèn)題及解決

    這篇文章主要介紹了mysql中文顯示為問(wèn)號(hào)?的問(wèn)題及解決方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-01-01
  • MYSQL必知必會(huì)讀書筆記第六章之過(guò)濾數(shù)據(jù)

    MYSQL必知必會(huì)讀書筆記第六章之過(guò)濾數(shù)據(jù)

    本文給大家分享MYSQL必知必會(huì)讀書筆記第六章之過(guò)濾數(shù)據(jù)的相關(guān)知識(shí),非常實(shí)用,特此分享到腳本之家平臺(tái),供大家參考
    2016-05-05
  • Xampp中mysql無(wú)法啟動(dòng)問(wèn)題的解決方法

    Xampp中mysql無(wú)法啟動(dòng)問(wèn)題的解決方法

    最近有朋友反映遇到了xampp的mysql啟動(dòng)不了這種情況,該怎么辦呢,所以下面這篇文章主要給大家介紹了關(guān)于Xampp中mysql無(wú)法啟動(dòng)問(wèn)題的解決方法,文中通過(guò)圖文介紹的非常詳細(xì),需要的朋友可以參考下
    2022-06-06
  • mysql?窗口函數(shù)?ROW_NUMBER、NTILE詳解

    mysql?窗口函數(shù)?ROW_NUMBER、NTILE詳解

    這篇文章主要介紹了mysql?窗口函數(shù)?ROW_NUMBER、NTILE,本文通過(guò)sql語(yǔ)句給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-07-07
  • MySQL中登錄與退出超全圖文講解

    MySQL中登錄與退出超全圖文講解

    大多數(shù)人都認(rèn)為在不需要事務(wù)化處理的情況下,MySQL是管理內(nèi)容最好的選擇,下面這篇文章主要給大家介紹了關(guān)于MySQL中登錄與退出圖文講解的相關(guān)資料,文中通過(guò)圖文介紹的非常詳細(xì),需要的朋友可以參考下
    2023-06-06
  • 淺析MySQL如何實(shí)現(xiàn)事務(wù)隔離

    淺析MySQL如何實(shí)現(xiàn)事務(wù)隔離

    使用過(guò)關(guān)系型數(shù)據(jù)庫(kù)的,應(yīng)該都事務(wù)的概念有所了解,知道事務(wù)有 ACID 四個(gè)基本屬性:原子性(Atomicity)、一致性(Consistency)、隔離性(Isolation)和持久性(Durability),今天我們主要來(lái)理解一下事務(wù)的隔離性
    2021-06-06
  • 使用sql語(yǔ)句insert之前判斷是否已存在記錄

    使用sql語(yǔ)句insert之前判斷是否已存在記錄

    這篇文章主要介紹了使用sql語(yǔ)句insert之前判斷是否已存在記錄,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-12-12

最新評(píng)論