设为首页 收藏本站
查看: 794|回复: 0

[经验分享] SQL Server-外部联接基础回顾(十三)

[复制链接]

尚未签到

发表于 2017-7-13 19:34:08 | 显示全部楼层 |阅读模式
前言
  本节我们继续讲讲联接类型中的外部联接,本节之后我们将讲述有关联接性能以及更深入的知识,简短内容,深入的理解,Always to review the basics。

外部联接
  外部联接又分为左外部联接和右外部联接,使用关键字分别是LEFT OUTER JOIN、RIGHT OUTER JOIN、FULL OUTER JOIN,在这里OUTER关键字时可选的。LEFT关键字表示保留左侧的行,RIGHT关键字表示保留右侧的行,FULL关键字表示左侧和右侧的行都保留。外部联接的第三个逻辑查询处理阶段识别保留表中基于ON谓词未能与另一个表匹配的行,此阶段添加这些行到前两个联接阶段生成的结果中,在这些外部行中,对于联接非保留侧的属性将使用NULL作为占位符。说了这么多,左外部联接就是以左表为基准,若右表满足条件则返回右侧的行,若不满足则返回NULL而非右侧的实际行数据,右外部联接同理。我们来看如下一个简单的例子



USE TSQL2012
GO
SELECT C.custid, C.companyname, O.orderid
FROM Sales.Customers AS C
LEFT JOIN Sales.Orders AS O ON C.custid = O.custid
DSC0000.png

  从上知,Sales.Customers表中的有一个客户没有任何订单,它的订单Id为22,通过左侧客户Id为22而右侧得到订单Id为NULL而得知。

超越外部联接基础知识
  通过上述对外部联接的介绍,我们知道通过外部联接能够得到缺失值,也就是不满足条件则返回NULL。这里我们假设有如下一场景,我们需要查询Orders订单表中所有订单,要求确保在范围2006年1月1日至2008年12月31日中每天至少有一行输出,对于范围能具有的订单的日期不做任何操作,但希望输出中包含没有订单的日期,使用NULL标记作为订单属性占位符。我们第一步需要解决的是得到2006年1月1日至2008年12月31日的所有日期,上一篇我们讲过交叉联接,通过交叉联接我们生成了数字表,这个时候就派上用场了。
  首先需要得到2006年1月1日至2008年12月31日之间间隔的天数,然后得到此间隔中的所有日期,如下:



USE TSQL2012
GO
SELECT DATEADD(DAY, n -1 ,'20060101') AS orderdate
FROM dbo.Nums
WHERE n <= DATEDIFF(DAY, '20060101','20081231') + 1
ORDER BY orderdate
DSC0001.png

  接下来通过上述得到的连续日期与Sales.Orders表中的订单日期进行匹配,从而得到订单所有信息,利用左外部联接不满足条件则返回NULL。



USE TSQL2012
GO
SELECT DATEADD(DAY, n -1 ,'20060101') AS orderdate,O.orderid,
O.custid, O.empid
FROM dbo.Nums
LEFT JOIN Sales.Orders AS O
ON DATEADD(DAY, dbo.Nums.n - 1, '20060101') = O.orderdate
WHERE dbo.Nums.n <= DATEDIFF(DAY, '20060101','20081231') + 1
ORDER BY orderdate
DSC0002.png


外部联接注意事项 (1)
  (1)WHERE造成外部联接成为逻辑上的内部联接
  我们看上述图中在订单日期等于2006-11-09时,此时orderid为NULL,此时我们首先来进行如下查询



USE TSQL2012
GO
SELECT orderdate, orderid, custid, empid
FROM Sales.Orders
WHERE orderdate > '20061108'
  我们查询Sales.Orders表中订单日期大于2006-11-08订单信息,我们看下返回结果
DSC0003.png

  此时我们发现订单日期为2006-11-09的订单信息没有,为何如此呢,因为利用WHERE子句它会过滤掉UNKNOWN即NULL值,为什么要讲这个呢,因为在外部联接中不满足条件的右表原本会返回NULL,但是若存在WHERE子句时,此时会导致所有外部行会被过滤掉,换句话说就是抵消了外部联接实际上在逻辑上就相当于是一个内部联接,这就在无意中造成了逻辑上的BUG。所以基于此我们可以得出如下结论:
  结论:在外部联接中若利用WHERE子句会过滤掉NULL,会导致所有外部行被过滤掉,实际上会抵消外部联接,最终外部联接在逻辑上就成为了一个内部联接。
  (2)多表联接造成外部联接成为逻辑上的内部联接
  在进行多联接时比如说首先进行两个表的外部联接,紧接着后面跟了一个内部联接的第三个表,如果内部联接中子句的谓词对自外部联接非保留侧的属性和来自第三个表的属性进行比较,此时所有外部行都会被过滤掉。是什么意思呢,当利用外部联接时会可能返回外部行的NULL值,此时再利用内部联接,因为NULL与任何值进行比较都会生成UNKNOWN,所以此时UNKNOWN会被ON筛选过滤掉。同上也就抵消外部联接,造成外部联接成为逻辑上的内部联接。如下



USE TSQL2012
GO
SELECT C.custid, O.orderid, OD.productid, OD.qty
FROM Sales.Customers AS C
LEFT JOIN Sales.Orders AS O ON O.custid = C.custid
INNER JOIN Sales.OrderDetails AS OD ON OD.orderid = O.orderid
DSC0004.png

  一般来说,无论何种类型的外部联接后跟一个内部联接或是外部联接子查询,外部行都会被过滤掉,当然,这是假设联接条件对来自左侧的NULL标记和右侧的任意值进行比较。为了解决这种问题,我们可以通过如下三种方案来解决。
  【1】将第二个内部联接替换为左外部联接。



USE TSQL2012
GO
SELECT C.custid, O.orderid, OD.productid, OD.qty
FROM Sales.Customers AS C
LEFT JOIN Sales.Orders AS O ON O.custid = C.custid
LEFT JOIN Sales.OrderDetails AS OD ON OD.orderid = O.orderid
DSC0005.png

  【2】首先使用内部联接,然后再使用右外部联接。



SELECT C.custid, O.orderid, OD.productid, OD.qty
FROM Sales.Orders AS O
LEFT JOIN Sales.OrderDetails AS OD ON OD.orderid = O.orderid
INNER JOIN Sales.Customers AS C ON O.custid = C.custid
DSC0006.png

  【3】在原有基础上改变,将内部联接变成一个独立的逻辑阶段



USE TSQL2012
GO
SELECT C.custid, O.orderid, OD.productid, OD.qty
FROM Sales.Customers AS C
LEFT JOIN
(Sales.Orders AS O INNER JOIN Sales.OrderDetails AS OD ON O.orderid = OD.orderid)
ON C.custid = O.custid
外部联接注意事项 (2)
  在外部联接中使用COUNT(*)进行聚合时,它会考虑内部行和外部行,因为它会计算行数而不管它们的内容,如下:



USE TSQL2012
GO
SELECT C.custid, COUNT(*) AS numorders
FROM Sales.Customers AS C
LEFT JOIN  Sales.Orders AS O ON O.custid = C.custid
GROUP BY C.custid
DSC0007.png

  由之前所查得知客户Id为22的orderid为NULL,即没有订单数量,所以这里就产生了BUG,因为COUNT(*)会包括NULL值,所以这里我们需要替换为COUNT(列名)。



USE TSQL2012
GO
SELECT C.custid, COUNT(O.orderid) AS numorders
FROM Sales.Customers AS C
LEFT JOIN  Sales.Orders AS O ON O.custid = C.custid
GROUP BY C.custid
DSC0008.png


总结
  本节我们重点讲述了外部联接基本知识以及注意事项,我们下节将讲述联接综合知识,简短的内容,深入的理解,我们下节再会,good night。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-393561-1-1.html 上篇帖子: 关于SQL Server 安装程序在运行 Windows Installer 文件时遇到错误 下篇帖子: SQL Server(四)——查询练习(45道习题)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表