【译注:此文为翻译,由于本人水平所限,疏漏在所难免,欢迎探讨指正】
原文链接:传送门。
使用OUTER APPLY 操作符
OUTER APPLY操作符工作起来和CROSS APPLY比较类似。唯一的不同在于OUTER APPLY操作符对于那些从表值函数不返回任何数据行的列仍旧会返回结果集。为了看其是如何工作的我们可以查看列表3的代码。
USE tempdb; GO SELECT * FROM dbo.SearchString AS S OUTER APPLY dbo.FindProductLike(S.String);
列表3:OUTER APPLY示例
当我运行列表3的代码,将得到报告2的输出结果。
ID String ProductName Price ----- ---------------- -------------------------- --------- 1 Red Red Santa Suit 199.99 1 Red Red Bells 49.99 2 Lights LED Lights 6.99 3 Star NULL NULL
报告2:通过运行列表3的代码产生的输出
通过使用OUTER APPLY操作符,你可以看到对于字符串值“Star”,存在一个数据行,其ProductName列和Price列均为Null值。这是因为SearchString表的字符串值“Star”并没有发现是Product表的任何ProductName列的一部分。当表值函数dbo.FindProductLike使用了列值而返回了一个空的数据行集时,SQL SERVER便会对表值函数产生的数据列赋以null值。OUTER APPLY操作符的这个功能类似与当在两个表之间使用OUTER JOIN而并没有匹配行的情形。当一个字符串传递给表值函数dbo.FindProductLike并确实返回数据行时,那么SearchString表的数据行便会与表值函数的结果进行关联,如果一个CROSS APPLY操作符被使用的时候发生的一样。
使用一个表值表达式
到此为止,我仅仅向你展示了APPLY操作符应用在一个数据集和一个表值函数之间的情形。除此之外,APPLY操作符同样可以与表值表达式一起工作。为了看这是如何进行的,我们可以查看列表4的代码:
,
列表4:CROSS APPLY与表值表达式一起使用
当我运行列表4的代码将得到报告3的输出。
ID String ProductName Price ----- ---------------- -------------------------- --------- 1 Red Red Santa Suit 199.99 1 Red Red Bells 49.99 2 Lights LED Lights 6.99
报告3:通过运行列表4的代码产生的输出
列表4的代码与列表2的代码是等效的,并且报告3的输出是与报告1的输出完全一致。在列表4中,我将CROSS APPLY右边的表达式字符串更改为相关的子查询,这个子查询使用了从SearchString表来的String列的值来找到其ProductName列包含此字符串值的所有Product数据。
同样我也可以拿到列表4的代码并通过将CROSS APPLY更改为OUTER APPLY从而使其与列表3的代码相等。我将让你来做这个更改并运行新的代码来验证它确实产生了和报告2一样的输出。
总结
APPLY操作符允许你将一个数据集中的列与一个表值函数或者一个表值表达式进行关联,通过使用APPLY操作符你可以写基于集合的查询,并且其中的一个集合是表值函数或者表值表达式。下次你需要将一个数据集的行与一个表值函数或者表值表达式进行关联,你可以使用APPLY操作符。
Q&A
您可以通过回答以下问题,来检验对APPLY 操作符的了解使用情况。
Q1:
APPLY 操作符允许您将记录集合与标量函数关联(TRUE 或 FALSE)?
- TRUE
- FALSE
Q2:
APPLY操作符支持哪两种格式?
- CROSS APPLY
- FULL APPLY
- INNER APPLY
- OUTER APPLY
Q3:
如果您想通过使用APPLY操作符从记录集合结合表值函数返回所有记录,APPLY操作符支持哪种格式?
- CROSS APPLY
- OUTER APPLY
A:
Q1:
正确答案是b或者False。APPLY操作符不允许您将记录集合与标量函数关联。但它允许您将记录集合与表值函数或标志表达式关联。
Q2:
正确答案是a和b。APPLY操作符支持CROSS APPLY和FULL APPLY。
Q3:
正确答案是b。a不正确的原因是如果您使用CROSS APPLY操作符您将不会返回所有行通过记录集合结合表值函数。OUTER APPLY操作是唯一格式,确保所有与表值函数关联的记录返回在最终的结果集合中。