mysql left join的基本用法以及on與where的區(qū)別
前言
我們在寫sql語句的時(shí)候,總是無法避免使用到連接關(guān)鍵詞,比如內(nèi)連接、外連接。種類是很多的,我在這里貼上一張?jiān)趧e處找到的圖:

這張圖我認(rèn)為是非常詳細(xì)了,它展示出了SQL語句中常見的鏈接類型,以本文中的left join為例,網(wǎng)上是這么給定義的:LEFT JOIN 關(guān)鍵字會從左表 那里返回所有的行,即使在右表中沒有匹配的行。
其實(shí)光從字面意思上來說的話,left join是比較好理解的,但是在使用的過程中,還是會有一些問題的,比如條件在on后面與在where后面,他們的結(jié)果是完全不一樣的,接下來我們就從淺到深去了解下left join。
實(shí)例
CREATE TABLE `class` ( `class_id` int NOT NULL, `class_name` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci DEFAULT NULL, `class_grade` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci DEFAULT NULL, PRIMARY KEY (`class_id`) USING BTREE ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb3 ROW_FORMAT=COMPACT;
score表:
CREATE TABLE `score` ( `class_id` int NOT NULL, `stu_id` varchar(11) CHARACTER SET utf8 COLLATE utf8_general_ci DEFAULT NULL, `score` int DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb3 ROW_FORMAT=COMPACT;
他們各有數(shù)據(jù):


Q1:select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on c.class_id=s.class_id。
我們對這個(gè)語句進(jìn)行分析:左表是class表,右表是score表,兩表的class_id為兩表關(guān)聯(lián)字段。當(dāng)左表class_id為1時(shí),右表有兩條記錄的class_id為1;當(dāng)左表class_id為2時(shí),右表有兩條記錄的class_id為2;當(dāng)左表class_id為3時(shí),右表有一條記錄的class_id為3,所以我們應(yīng)當(dāng)?shù)玫轿鍡l記錄:
1 語文 A A002 82 1 語文 A A001 91 2 數(shù)學(xué) B A002 87 2 數(shù)學(xué) B A001 95 3 英語 C B003 65
Q2:select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on c.class_id=s.class_id and s.score=90。
以第一題的例子為基礎(chǔ),這個(gè)又多了個(gè)s.score=90的條件,這個(gè)表示右表score的score字段為90,但是我們看表的數(shù)據(jù),發(fā)現(xiàn)右表沒有score為90,ok,那結(jié)果會不會是空呢?畢竟右表是沒有符合條件的數(shù)據(jù)的。
事實(shí)上,如果執(zhí)行這條sql語句的話,最終是可以得到結(jié)果的,只不過只會得到三條數(shù)據(jù),左表中的字段全部顯示,右表中的字段全部對應(yīng)為空,這是因?yàn)橛冶碇袥]有記錄的score是90,所以右表的字段才是空的。
on是先對數(shù)據(jù)進(jìn)行過濾,然后再進(jìn)行連接,而where是在兩表進(jìn)行關(guān)聯(lián)查詢之后,再進(jìn)行過濾的,on與where的區(qū)別就在這里,之后的例子也會涉及這兩個(gè)區(qū)別的。
1 語文 A 2 數(shù)學(xué) B 3 英語 C
Q3:select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on c.class_id=s.class_id where c.class_name=‘語文’ and s.score=90。
這個(gè)語句里面就涉及到了where關(guān)鍵字了,因此是需要注意與on的不同。這個(gè)sql的查詢結(jié)果為空,這是因?yàn)橄葘杀磉B接查詢出結(jié)果,然后再進(jìn)行過濾,而連表查詢出的結(jié)果是沒有記錄符合class_name=‘語文’,并且score=90,所以查詢結(jié)果為空。
Q4:select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on c.class_id=s.class_id and 1=0。
我們有時(shí)在sql的where條件后面會加上1=1,而這里的1=0則是表示兩表關(guān)聯(lián)失敗,所以這個(gè)的結(jié)果只會顯示出左表的數(shù)據(jù)。
1 語文 A null null 2 數(shù)學(xué) B null null 3 英語 C null null
Q5:select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on 1=0。
這個(gè)與上面的sql語句的執(zhí)行結(jié)果是一樣的,其實(shí)都是將左表的內(nèi)容全部顯示。
1 語文 A null null 2 數(shù)學(xué) B null null 3 英語 C null null
Q6:select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on c.class_id=s.class_id and c.class_name=‘語文’。
這個(gè)需要在右表中過濾出class_name為語文的記錄,有兩條,然后左表去連接這兩條記錄,所以不難看出,有四條結(jié)果:
1 語文 A A002 82 1 語文 A A001 91 2 數(shù)學(xué) B null null 3 英語 C null null
Q7:select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on c.class_id=s.class_id and c.class_name=‘英語’。
分析同上一題(空白的地方的結(jié)果為null):
1 語文 A 2 數(shù)學(xué) B 3 英語 C B003 65
Q8:select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on c.class_id=s.class_id and c.class_name=‘體育’。
右表中是沒有數(shù)據(jù)的class_name為體育的,所以右表為空,顯示左表的全部數(shù)據(jù),右表對應(yīng)的字段為空:
1 語文 A 2 數(shù)學(xué) B 3 英語 C
Q9:select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on c.class_id=s.class_id and c.class_name=‘語文’ and s.score=91。
右表中只有一條記錄的score為91,所以需要拿左表與右表的這一條數(shù)據(jù)進(jìn)行關(guān)聯(lián),左表只有語文可以與右表的那一條數(shù)據(jù)對上,所以結(jié)果為:
1 語文 A A001 91 2 數(shù)學(xué) B 3 英語 C
Q10:select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on c.class_id=s.class_id and c.class_name=‘體育’ and s.score=90。
右表中沒有數(shù)據(jù)的score為90,同樣左表中也沒有class_name為體育,但是這并不意味著最后的結(jié)果就是空了,只要沒有where條件,最終的結(jié)果數(shù)量最起碼也會是左表中原先的數(shù)據(jù)數(shù)量,所以這條sql會返回左表的全部數(shù)據(jù)。
1 語文 A 2 數(shù)學(xué) B 3 英語 C
Q11:select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on c.class_id=s.class_id where c.class_name=‘英語’。
需要注意的點(diǎn):條件是在where中的,也就是在表關(guān)聯(lián)之后,再進(jìn)行過濾的,所以最終的結(jié)果只會有一條:
3 英語 C B003 65
Q12:select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on c.class_id=s.class_id where s.score=91。
和上面一樣,是在連表查詢之后,找出score=90的數(shù)據(jù):
1 語文 A A001 91
Q12;select c.class_id,c.class_name,c.class_grade,s.stu_id,s.score from class c left join score s on c.class_id=s.class_id where c.class_name=‘語文’ and s.score=91。
我們將兩表連接查詢后,找出結(jié)果中class_name為語文,score為91的記錄,只有一條:
1 語文 A A001 91
總結(jié)
通過這12道sql語句,其實(shí)就可以基本上掌握left join的使用以及可能出現(xiàn)的場景了,尤其是on與where在執(zhí)行sql語句時(shí),條件位置不同,最終的結(jié)果也不大相同。我覺得需要記住,只要沒有where條件,那么最終結(jié)果的數(shù)量最起碼與左表的數(shù)據(jù)數(shù)量相同,不可能會出現(xiàn)最終的結(jié)果為空的情況。
相關(guān)文章
MySQL聯(lián)合索引與最左匹配原則的實(shí)現(xiàn)
最左匹配原則在我們MySQL開發(fā)過程中和面試過程中經(jīng)常遇到,為了加深印象和理解,我在這里把MySQL的最左匹配原則詳細(xì)的講解一下,感興趣的可以了解一下2023-12-12
Windows系統(tǒng)中配置開啟MySQL數(shù)據(jù)庫日志的完整步驟
這篇文章主要給大家介紹了關(guān)于Windows系統(tǒng)中配置開啟MySQL數(shù)據(jù)庫日志的完整步驟,大家要開啟MySQL數(shù)據(jù)庫的日志功能,可以按照本文介紹的步驟進(jìn)行操作,需要的朋友可以參考下2023-09-09
linux下mysql開啟遠(yuǎn)程訪問權(quán)限 防火墻開放3306端口
這篇文章主要為大家詳細(xì)介紹了linux下mysql開啟遠(yuǎn)程訪問權(quán)限,防火墻開放3306端口,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01
mysql中復(fù)制表結(jié)構(gòu)的方法小結(jié)
這篇文章主要介紹了mysql中復(fù)制表結(jié)構(gòu)的方法,需要的朋友可以參考下2014-07-07
Mysql中校對集utf8_unicode_ci與utf8_general_ci的區(qū)別說明
一直對utf8_unicode_ci與utf8_general_ci這2個(gè)校對集很迷惑,今天查了手冊有了點(diǎn)眉目。不過對中文字符集來說采用utf8_unicode_ci與utf8_general_ci時(shí)有何區(qū)別還是不清楚2012-03-03
mysql實(shí)現(xiàn)查詢結(jié)果導(dǎo)出csv文件及導(dǎo)入csv文件到數(shù)據(jù)庫操作
這篇文章主要介紹了mysql實(shí)現(xiàn)查詢結(jié)果導(dǎo)出csv文件及導(dǎo)入csv文件到數(shù)據(jù)庫操作,結(jié)合實(shí)例形式分析了mysql相關(guān)數(shù)據(jù)庫導(dǎo)出、導(dǎo)入語句使用方法及操作注意事項(xiàng),需要的朋友可以參考下2018-07-07
mysql觸發(fā)器實(shí)現(xiàn)oracle物化視圖示例代碼
mysql觸發(fā)器實(shí)現(xiàn)oracle物化視圖即不是基于基表的虛表,而是根據(jù)表實(shí)際存在的實(shí)表,需要的朋友可以參考下2014-02-02

