SQL Server中CROSS APPLY和OUTER APPLY应用

SQL Server中CROSS APPLY和OUTER APPLY应用

1.Cross Apply和Outer Apply的理解

新增的APPLY表运算符把右表表达式应用到左表表达式中的每一行。
它不像JOIN那样先计算哪个表表达式都可以,APPLY必选先逻辑地计算左表达式。这种计算输入的逻辑顺序允许把右表达式关联到左表表达式。

APPLY有两种形式,一个是OUTER APPLY,一个是CROSS APPLY,CROSS APPLY仅返回外部表中通过表值函数生成结果集的行。OUTER APPLY既返回生成结果集的行,也返回不生成结果集的行,其中表值函数生成的列中的值为 NULL。

语法:
APPLY 操作符作用于 FROM 字句中,它的具体语法如下:

FROM left_table_source    
  { CROSS | OUTER } APPLY right_table_source

在上面的语法中,left_table_source 表示坐输入;right_table_source 表示右输入,一般是表值函数,但亦可是视图、表、派生表。

2.示例测试

(1).CROSS APPLY

说明:只返回外部表中包含的数据行,这些航从表值函数中生成结果集。 比如:LargeTable表中的某一列存储的数据是以“:”号分隔的数据,我们处理的时候,可能要先把这个值,先分隔,然后把分隔后的每个值单独一行放在一张表中,然后对这个表做处理。原始数据(LargeTable表):


为了简单,先用其中id=2的一行处理,这些以:号分隔的数据,可能是我们某张表的主键(t1),我们可能需要把这些数值提出来,放在一张临时表中,和t1表关联,做一些处理。处理分隔数据效果图:


如果用之前的版本处理这个操作的话,应该复杂些。这只是用其中一行做的处理,如果我们用上图的3行都做这样处理,把三行以:号分隔的数值都放在一个表中,该怎么处理呢?
用APPLY表运算符一行语句就能简单的处理以上操作:

SELECT a FROM dbo.LargeTable AS LT    --实际表 
    CROSS APPLY dbo.split(LT.Name,':')    --自定义表值函数,处理以某个字符分隔的数据,把这些数据,返回一张表 
WHERE a <> ''                 --去掉结果表中a字段为空的数据 </pre>

执行结果:


需要额外定义的就是那个自定义表值函数(Split),类似.Net中Split操作,代码如下:

CREATE Function [dbo].[Split] ( @Sql varchar(8000), @Splits varchar(10)) 
returns @temp Table (a varchar(100)) As
Begin
    Declare @i Int
    Set @Sql = RTrim(LTrim(@Sql)) 
    Set @i = CharIndex(@Splits,@Sql) While @i >= 1
    Begin
        Insert @temp Values(Left(@Sql,@i-1)) 
        Set @Sql = SubString(@Sql,@i+1,Len(@Sql)-@i) 
        Set @i = CharIndex(@Splits,@Sql) 
    End

    If @Sql <> ''
        Insert @temp Values (@Sql) 
        Return
End

------------调用示例-------------

SELECT * FROM dbo.Split('581::579::519::279::406::361::560',':')

(2).OUTER APPLY

说明:无论联接是否存在,都会从外部表和表值函数中返回数据行。这与OUTER JOIN 很类似。如果表值函数中没有行存在,则从函数中返回的列值为NULL。
场景描述:有个供货商表(Supplier)和供货商产品表(Products),我们要取每一个供货商中单价最高的两个产品。其表数据:


我们创建一个自定义表值函数(dbo.fn_top_products),该函数根据供货商ID返回单价最高的两个商品。下面用OUTER APPLY形式来查询,看会出现什么结果:

IF OBJECT_ID('dbo.fn_top_products') IS NOT NULL  
DROP FUNCTION dbo.fn_top_products; 

GO 
--根据供货商ID获得单价最高的两件商品
CREATE FUNCTION dbo.fn_top_products  (@supid AS INT) 
RETURNS TABLE  
AS  
RETURN  
SELECT TOP(2) Id AS ProductId,ProductName,UnitPrice 
FROM dbo.Products 
WHERE SupplierId = @supid  
ORDER BY UnitPrice DESC  
GO 

--执行以下语句:

SELECT S.id AS SupplierId,S.CompanyName,UnitPrice 
FROM dbo.Supplier AS S 
OUTER APPLY dbo.fn_top_products(S.id) AS P 

执行结果:


注意最后为NULL的记录,reed公司因为没有商品,所以单价为NULL了。

如果用CROSS APPLY形式,执行以下查询:

SELECT S.id AS SupplierId,S.CompanyName,UnitPrice 
FROM dbo.Supplier AS S 
CROSS APPLY dbo.fn_top_products(S.id) AS P 

执行结果:


这样是不是OUTER APPLY和CROSS APPLY的区别更清晰了。

APPLY的执行过程,它先逻辑计算左表表达式(以上的LargeTable表),然后把右表达式(以上的自定义表值函数Split)应用到左表表达式的每一行。实际是把外部查询的列引用作为参数传递给表值函数。

4.参考博客:
【俊哥儿张】MS SQL Server:APPLY 运算符详解
SQL Server中CROSS APPLY和OUTER APPLY的应用详解

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,905评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,140评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,791评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,483评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,476评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,516评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,905评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,560评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,778评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,557评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,635评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,338评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,925评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,898评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,142评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,818评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,347评论 2 342

推荐阅读更多精彩内容

  • 第5章 引用类型(返回首页) 本章内容 使用对象 创建并操作数组 理解基本的JavaScript类型 使用基本类型...
    大学一百阅读 3,204评论 0 4
  •   引用类型的值(对象)是引用类型的一个实例。   在 ECMAscript 中,引用类型是一种数据结构,用于将数...
    霜天晓阅读 1,034评论 0 1
  • MySQL技术内幕:SQL编程 姜承尧 第1章 SQL编程 >> B是由MySQL创始人之一Monty分支的一个版...
    沉默剑士阅读 2,402评论 0 3
  • 函数和对象 1、函数 1.1 函数概述 函数对于任何一门语言来说都是核心的概念。通过函数可以封装任意多条语句,而且...
    道无虚阅读 4,521评论 0 5
  • 本章内容 使用对象 创建并操作数组 理解基本的 JavaScript 类型 使用基本类型和基本包装类型 引用类型的...
    闷油瓶小张阅读 674评论 0 0