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

關(guān)于join?on和where執(zhí)行順序分析

 更新時(shí)間:2023年03月06日 10:53:39   作者:波波仔86  
這篇文章主要介紹了join?on和where執(zhí)行順序,如果是inner?join,?放on和放where產(chǎn)生的結(jié)果一樣,?執(zhí)行計(jì)劃也是一樣,但推薦使用on,本文對(duì)join?on和where執(zhí)行順序給大家詳細(xì)講解,需要的朋友可以參考下

join on和where執(zhí)行順序

1、join中相比where優(yōu)先推薦on

WHERE子句中使用的連接語句,在數(shù)據(jù)庫語言中,被稱為隱性連接。INNER JOIN……ON子句產(chǎn)生的連接稱為顯性連接。(其他JOIN參數(shù)也是顯性連接)WHERE和INNER JOIN產(chǎn)生的連接關(guān)系,沒有本質(zhì)區(qū)別,結(jié)果也一樣。但是!隱性連接隨著數(shù)據(jù)庫語言的規(guī)范和發(fā)展,已經(jīng)逐漸被淘汰,比較新的數(shù)據(jù)庫語言基本上已經(jīng)拋棄了隱性連接,全部采用顯性連接了。

2. 邏輯上一個(gè)query的執(zhí)行順序(不是實(shí)際) 

1. FROM 
2. ON 
3. JOIN 
4. WHERE 
5. GROUP BY 
6. WITH CUBE or WITH ROLLUP 
7. HAVING 
8. SELECT 
9. DISTINCT 
10. ORDER BY 
11. TOP 
說是“邏輯上” 順序,因?yàn)閷?shí)際執(zhí)行時(shí)還要看索引,數(shù)據(jù)分布等,看最終優(yōu)化器如何處理,最真實(shí)的順序肯定是執(zhí)行計(jì)劃展示的順序。

SQL語句中join連表時(shí)on和where后都可以跟條件,那么對(duì)查詢結(jié)果集,執(zhí)行順序,效率是如何呢? 通過查詢資料發(fā)現(xiàn): 
區(qū)別: 
on是對(duì)中間結(jié)果進(jìn)行篩選,where是對(duì)最終結(jié)果篩選。 

執(zhí)行順序: 
先進(jìn)行on的過濾, 而后才進(jìn)行join。 

效率: 
如果是inner join, 放on和放where產(chǎn)生的結(jié)果一樣, 但沒說哪個(gè)效率速度更高? 如果有outer join (left or right), 就有區(qū)別了, 因?yàn)閛n生效在先, 已經(jīng)提前過濾了一部分?jǐn)?shù)據(jù), 而where生效在后. 
最后來了解下T-SQL對(duì)查詢邏輯處理。 
T-SQL邏輯查詢的各個(gè)階段(編號(hào)代表順序): 
(5)SELECT DISTINCT TOP(<top_specification>) <select_list>                      
(1)FROM <left_table> <join_type> JOIN <right_table> ON <on_predicate> 
(2)WHERE <where_predicate> 
(3)GROUP BY <group_by_specification> 
(4)HAVING <having_predicate> 
(6)ORDER BY <order_by_list> 

T-SQL在查詢各個(gè)階級(jí)分別干了什么: 
(1)FROM 階段 
    FROM階段標(biāo)識(shí)出查詢的來源表,并處理表運(yùn)算符。在涉及到聯(lián)接運(yùn)算的查詢中(各種join),主要有以下幾個(gè)步驟: 
  a.求笛卡爾積。不論是什么類型的聯(lián)接運(yùn)算,首先都是執(zhí)行交叉連接(cross join),求笛卡兒積,生成虛擬表VT1-J1。 

      b.ON篩選器。這個(gè)階段對(duì)上個(gè)步驟生成的VT1-J1進(jìn)行篩選,根據(jù)ON子句中出現(xiàn)的謂詞進(jìn)行篩選,讓謂詞取值為true的行通過了考驗(yàn),插入到VT1-J2。 

     c.添加外部行。如果指定了outer join,還需要將VT1-J2中沒有找到匹配的行,作為外部行添加到VT1-J2中,生成VT1-J3。 

    經(jīng)過以上步驟,F(xiàn)ROM階段就完成了。概括地講,F(xiàn)ROM階段就是進(jìn)行預(yù)處理的,根據(jù)提供的運(yùn)算符對(duì)語句中提到的各個(gè)表進(jìn)行處理(除了join,還有apply,pivot,unpivot) 

(2)WHERE階段 
     WHERE階段是根據(jù)<where_predicate>中條件對(duì)VT1中的行進(jìn)行篩選,讓條件成立的行才會(huì)插入到VT2中。 

(3)GROUP BY階段 
      GROUP階段按照指定的列名列表,將VT2中的行進(jìn)行分組,生成VT3。最后每個(gè)分組只有一行。 

(4)HAVING階段 
      該階段根據(jù)HAVING子句中出現(xiàn)的謂詞對(duì)VT3的分組進(jìn)行篩選,并將符合條件的組插入到VT4中。 

(5)SELECT階段 
  這個(gè)階段是投影的過程,處理SELECT子句提到的元素,產(chǎn)生VT5。這個(gè)步驟一般按下列順序進(jìn)行 
        a.計(jì)算SELECT列表中的表達(dá)式,生成VT5-1。 
        b.若有DISTINCT,則刪除VT5-1中的重復(fù)行,生成VT5-2 
        c.若有TOP,則根據(jù)ORDER BY子句定義的邏輯順序,從VT5-2中選擇簽名指定數(shù)量或者百分比的行,生成VT5-3 

(6)ORDER BY階段 
     根據(jù)ORDER BY子句中指定的列明列表,對(duì)VT5-3中的行,進(jìn)行排序,生成游標(biāo)VC6.

如果是inner join, 放on和放where產(chǎn)生的結(jié)果一樣, 執(zhí)行計(jì)劃也是一樣,但推薦使用on。但如果有outer join (left or right), 就有區(qū)別了, 因?yàn)閛n生效在先, 已經(jīng)提前過濾了一部分?jǐn)?shù)據(jù), 而where生效在后,而且on對(duì)于outer join有不生效的情況,需要看and條件是作用在左表還是右表。

擴(kuò)展:SQL執(zhí)行順序join在where前面

  • 取a表和b表join的數(shù)據(jù)sql

原本意圖是取a表的昨日分區(qū)數(shù)據(jù)和b表的昨日分區(qū)數(shù)據(jù)進(jìn)行join,但是發(fā)現(xiàn)加上and b.dt = '${daily}'的條件后就取不到在a表中存在的數(shù)據(jù)了

select substr(a.create_time,1,7) create_month,a.service_id,a.unit_number,a.company_name,coalesce(b.unified_social_credit_code,'unknown') unified_social_credit_code
,a.company_id,b.company_id as company_id_b,a.dt,b.dt
from clouduser.dz_company a 
left join clouduser.dz_company_detail b on a.company_id = b.company_id
where a.dt = '${daily}' 
--and b.dt = '${daily}'
and substr(a.create_time,1,7) = '2022-09'
and a.company_name = '浙江港都電子有限公司';

運(yùn)行結(jié)果

在這里插入圖片描述

  • 原因是where執(zhí)行是在join之后,join出來的臨時(shí)表中,由于在b表中沒有匹配到數(shù)據(jù),所以b.dt為null,這時(shí)候執(zhí)行where b.dt = '${daily}'的條件篩選不到這條數(shù)據(jù)。
  • 優(yōu)化方法

現(xiàn)在子查詢中將b表的昨日分區(qū)查詢出來,再和a表join

with
r1 as (select * from clouduser.dz_company_detail where dt = '2022-10-17'),
r2 as (
select substr(a.create_time,1,7) create_month,a.service_id,a.unit_number,a.company_name,coalesce(b.unified_social_credit_code,'unknown') unified_social_credit_code
,a.company_id,b.company_id as company_id_b
from clouduser.dz_company a 
left join r1 b on a.company_id = b.company_id
where a.dt = '${daily}' 
--and b.dt = '${daily}'
and substr(a.create_time,1,7) = '2022-09'
and a.company_name = '浙江港都電子有限公司'
)
select * from r2;

執(zhí)行結(jié)果

在這里插入圖片描述

可以看到正常查詢出a表中的數(shù)據(jù)了

到此這篇關(guān)于join on和where執(zhí)行順序的文章就介紹到這了,更多相關(guān)join on和where執(zhí)行順序內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • MySQL流程控制IF()、IFNULL()、NULLIF()、ISNULL()函數(shù)的使用

    MySQL流程控制IF()、IFNULL()、NULLIF()、ISNULL()函數(shù)的使用

    這篇文章介紹了MySQL流程控制IF()、IFNULL()、NULLIF()、ISNULL()函數(shù)的使用方法,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-12-12
  • Mysql中的Datetime和Timestamp比較

    Mysql中的Datetime和Timestamp比較

    這篇文章主要介紹了Mysql中的Datetime和Timestamp比較,本文總結(jié)了它們的相同點(diǎn)和不同點(diǎn)以及時(shí)間格式介紹等,需要的朋友可以參考下
    2015-03-03
  • InnoDB數(shù)據(jù)庫死鎖問題處理

    InnoDB數(shù)據(jù)庫死鎖問題處理

    本文給大家講解的是mysql數(shù)據(jù)庫InnoDB類型,在update表的時(shí)候出現(xiàn)死鎖現(xiàn)象的原因及解決辦法,有需要的小伙伴可以參考下。
    2016-03-03
  • Mysql字符串類型如何通過order by排序的正確方式

    Mysql字符串類型如何通過order by排序的正確方式

    這篇文章主要介紹了Mysql字符串類型如何通過order by排序的正確方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-08-08
  • MySQL與Oracle的語法區(qū)別詳細(xì)對(duì)比

    MySQL與Oracle的語法區(qū)別詳細(xì)對(duì)比

    Oracle和mysql的一些簡(jiǎn)單命令對(duì)比在本文中將會(huì)涉及到很多的實(shí)例,感興趣的你不妨學(xué)習(xí)一下,就當(dāng)鞏固自己的知識(shí)了
    2013-03-03
  • MySQL: mysql is not running but lock exists 的解決方法

    MySQL: mysql is not running but lock exists 的解決方法

    下面可以參考下面的方法步驟解決。最后查到一個(gè)網(wǎng)友說可能和log文件有關(guān),于是將log文件給移除了,再重啟MySQL終于OK了
    2009-06-06
  • 對(duì)MySQL慢查詢?nèi)罩具M(jìn)行分析的基本教程

    對(duì)MySQL慢查詢?nèi)罩具M(jìn)行分析的基本教程

    這篇文章主要介紹了對(duì)MySQL慢查詢?nèi)罩具M(jìn)行分析的基本教程,文中提到的Query-Digest-UI這個(gè)基于B/S的圖形化查看工具非常好用,需要的朋友可以參考下
    2015-12-12
  • mysql遠(yuǎn)程跨庫聯(lián)合查詢的示例

    mysql遠(yuǎn)程跨庫聯(lián)合查詢的示例

    本文主要介紹了mysql遠(yuǎn)程跨庫聯(lián)合查詢的示例,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • 詳解MySQL?substring()?字符串截取函數(shù)

    詳解MySQL?substring()?字符串截取函數(shù)

    MySQL 查詢數(shù)據(jù)有時(shí)候需要對(duì)數(shù)據(jù)項(xiàng)進(jìn)行日期格式化或截取特定部分的操作,當(dāng)需要對(duì)字符串進(jìn)行截取加工時(shí)用到了 substring() 函數(shù),這篇文章主要介紹了MySQL?substring()?字符串截取函數(shù),需要的朋友可以參考下
    2022-07-07
  • SQL實(shí)現(xiàn)LeetCode(196.刪除重復(fù)郵箱)

    SQL實(shí)現(xiàn)LeetCode(196.刪除重復(fù)郵箱)

    這篇文章主要介紹了SQL實(shí)現(xiàn)LeetCode(196.刪除重復(fù)郵箱),本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-08-08

最新評(píng)論