SQL語句匯總(終篇)—— 表聯接與聯接查詢


既然是最后一篇那就不能只列出些干枯的標准語句,更何況表聯接也是SQL中較難的部分,所以此次搭配題目來詳細闡述表聯接。

 

上一篇博文說到相關子查詢效率低下,那我們怎么能將不同表的信息一起查詢出來呢?這就需要用到表聯接。

和之前的UNION組合查詢不同,UNION是將不同的表組合起來,也就是縱向聯接,說白了就是豎着拼起來。

而表聯接是通過笛卡爾乘積將表進行橫向聯接,所謂的通過笛卡爾乘積簡單說就是兩表的行依次相聯再相加。要想更詳細的理解可以百度下,畢竟本文主要是匯總SQL語句。

現在有如下兩張表:

這是當初老師布置的一份作業,我偷個懶就不改數據了。不過把這些真神級人物的大名貼出來做“實驗”總覺得心里有很虛,更何況大部分都是IT業的。如有什么不敬我先道個歉,別跟我一般見識。

好了,扯遠了。怎么聯接這兩張表呢?標准寫法:

SELECT * FROM t_student JOIN t_class 

結果這里只截一小部分圖,因為笛卡爾乘積后的行數等於兩張表的行數乘積,實在太多了。

這里就可以理解表聯接的原理了,依次相連再相加。當然其中很多是無效行,為了去除無效的行我們就要用到外鍵來進行約束。學生表中的_fk與班級表中的_infor相關聯:

SELECT * FROM t_student s JOIN t_class c ON s._fk=c._infor; 

結果:

這里通過外鍵的匹配我們就得到了一張完美的聯接之后的表,它可以看做一張新表,想要任何數據均可以從此表中查詢,這就是表聯接的強大之處。

 

表聯接的分類:

內聯接:

內聯接是指兩個表中某一行相關的列值匹配時,這一行才會出現在表中。就像上例中s._fk與c._infor相同時才會出行該行,其他的行剔除。

語法為INNER JOIN 其中INNER可以省略。

內聯接的簡寫:

SELECT * FROM t_student s,t_class c WHERE c._infor = s._fk 

* 此寫法也是我們用的最多的。

 

外聯接:

分為左外聯接與右處聯接。

外聯接是指不管有沒有匹配,被定義了外聯接的表數據都要出現在結果中。比如左外聯接,那么在JOIN左邊的表就被定義為外聯接,那么此表中所有數據都會出現在查詢結果中。
注意班級表中的四班是沒有學生的,所以在內聯接之后理所當然的被剔除了。現在以外聯接做示例:
SELECT * FROM t_student s RIGHT JOIN t_class c ON s._fk=c._infor; 

上面SQL中表t_class在寫在JOIN的右邊,所以我們用RIGHT JOIN來進行外聯接。

最下面多了一行四班的信息

例如我們想查出還沒有學生錄入的班級信息:

SELECT c._id,c._cname,c._code FROM t_student s RIGHT JOIN t_class c ON s._fk=c._infor WHERE s._id IS NULL; 

這就是外聯接的用法,通常用在我們想要的數據匹配不上時。

 

自聯接:

自聯接屬於內聯接或外聯接的一種特例,自聯接所聯接的表均是來自同一張,用法個人感覺還是比較巧妙的。

現有一表如下:

表中,6個人均屬於某公司的員工。區別是李四為張三和王五的領導,張八為趙六和孫七的領導。leader_idwork_id相關聯。

現在可以通過自聯接巧妙的將一張表分為員工部分和領導部分:

SELECT w.work_name,l.work_name 領導姓名 FROM t_emp w,t_emp l WHERE w.leader_id=l.work_id;

注意別名的用法

結果:

是不是有點方便?

 

知識點羅列到這里,做題時間到:

1.查詢鳳姐所在的班級

SELECT _cname FROM t_student s,t_class c WHERE c._infor = s._fk AND s._name = '鳳姐';

2.查詢同朱軍同班級的學生

SELECT s._name FROM t_student s WHERE s._fk = (
        SELECT cc._infor FROM t_class cc,t_student ss WHERE ss._fk = cc._infor AND ss._name = '朱軍'
) AND s._name != '朱軍';

本題中,括號內為聯接后的表,其返回的是'朱軍'所在班級的_infor,然后主查詢在學生表中匹配與_infor相等的_fk的行,最后從匹配成功后的行中剔除'朱軍'自己。

3.查詢每個班級的人數

SELECT d._cname,COUNT(_name) FROM (SELECT ss.*,cc._cname FROM t_class cc LEFT JOIN t_student ss ON ss._fk = cc._infor) d GROUP BY d._cname;

本題中,括號內為班級表外聯接后的表,並給該聯接后的表以別名d,按d的班級名稱d._cname分組后統計各班人數。這里之所以用外聯接還是因為四班沒有學生但依然要統計。

4.查詢班級人數最多的班級

SELECT cc._cname,COUNT(_name) FROM t_class cc,t_student ss WHERE cc._infor = ss._fk GROUP BY cc._cname HAVING COUNT(_name) >=ALL(
    SELECT COUNT(_name) FROM t_class c,t_student s WHERE c._infor = s._fk GROUP BY c._cname
);

這個有點凶殘,用了兩次表聯接。括號內返回的是每個班的人數:

之后外部又使用了一次表聯接,將每個班的人數與括號內的返回值逐一比較,得到最大值,然后找到最大值所在的班級。這里就體現了對SQL執行順序的理解有多重要了,聯接、分組、過濾等等的先后順序。

結果:

5.查詢每個班中年齡最低的人

SELECT cc._cname,ss._name,ss._age FROM t_student ss,t_class cc WHERE ss._fk = cc._infor AND ss._age <=ALL(
    SELECT MIN(s._age) FROM t_student s WHERE ss._fk = s._fk
);

本題中,括號內部返回一個學生表中的最小年齡,外部進行表聯接后將年齡列對返回值進行比較,若小於等於返回的最小值那其本身也為最小值。

如果括號內部不加判斷條件WHERE ss._fk = s._fk,則最后只會查詢出一條年齡最小的數據,而並沒有按我們想要的查詢出每個班的最小值。

如:

有人會問了既然按班分,用分組不就好了?但要注意的是最小年齡的人不只一個,而分組后每一個班只會顯示一個人。所以這里用了關聯條件WHERE ss._fk = s._fk來讓內外表關聯,從而統計出所有我們想要的值。

結果:

 

這里終於把這個小系列寫完了,非常非常感謝各位的支持,接下來打算分享一些有趣實用的東西,希望大家繼續支持,謝謝!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM