SQL Server中CROSS APPLY连接操作

2024-06-18 02:44

本文主要是介绍SQL Server中CROSS APPLY连接操作,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在 SQL Server 中,CROSS APPLY 是一个连接操作,它类似于 INNER JOIN,但有一些关键差异,特别是在处理表值函数(TVF)、行集函数或子查询时。CROSS APPLY 返回对于外部查询中的每一行,在内部查询或函数中都存在的行。但是,如果内部查询或函数对于外部查询的某行没有返回任何行,则该外部查询的行不会出现在结果集中。

以下是一些关键点,用于理解 CROSS APPLY:

  • 与 INNER JOIN 的比较:虽然 CROSS APPLY 在某些情况下可能产生与 INNER JOIN 相同的结果,但它们在处理一对多关系时有所不同。特别是当内部查询或函数返回多行时,CROSS APPLY 会为外部查询的每一行与内部查询或函数的每一行组合生成结果。
  • 与子查询的比较:与在 SELECT 语句中使用的子查询不同,CROSS APPLY 中的子查询或函数可以引用外部查询的列,就像它们在 JOIN 条件中一样。
  • 表值函数:CROSS APPLY 经常与表值函数一起使用,特别是当这些函数返回基于外部查询行值的动态结果集时。
  • 性能:在某些情况下,CROSS APPLY 可能会比使用其他连接类型(如 INNER JOIN)的查询更快,因为它可以更早地过滤掉不满足条件的行。但是,性能总是取决于具体的查询和数据分布。

假设我们有两个表:T_Product 和 T_Product_imgs。每个产品可能有多个上传的图片,但我们想为每个产品选择其最新图片。这可以通过使用 CROSS APPLY 和一个子查询来实现,该子查询返回每个产品的最新图片。

CREATE TABLE T_Product
(ID BIGINT IDENTITY(1,1) CONSTRAINT PK_ProductID PRIMARY KEY,Title NVARCHAR(64) NOT NULL CONSTRAINT DF_ProductTitle DEFAULT N'',Tag NVARCHAR(64) NOT NULL CONSTRAINT DF_ProductTag DEFAULT N'',Price DECIMAL(18,6) NOT NULL,CreateUser NVARCHAR(32) NOT NULL,CreateDate DATETIME DEFAULT GETDATE()
)CREATE TABLE T_Product_imgs
(ID BIGINT IDENTITY(1,1) CONSTRAINT PK_Product_imgsID PRIMARY KEY,Pro_id BIGINT NOT NULL,imgName NVARCHAR(64),CreateUser NVARCHAR(32) NOT NULL,CreateDate DATETIME DEFAULT GETDATE()
)
ALTER TABLE T_Product_imgs ADD CONSTRAINT FK_Product_imgs_ID FOREIGN KEY (Pro_id) REFERENCES T_Product (id) INSERT INTO T_Product (Title,Tag,Price,CreateUser,CreateDate) VALUES(N'雷霆打码机',N'打码',900000.89,N'DBA卢飞虎','2020-08-08 23:20:56')
INSERT INTO T_Product (Title,Tag,Price,CreateUser,CreateDate) VALUES(N'大族钻机',N'钻孔',1050000.66,N'DBA卢飞虎','2020-09-01 23:20:56')
INSERT INTO T_Product (Title,Tag,Price,CreateUser,CreateDate) VALUES(N'雪龙锣机',N'锣板',530000.76,N'DBA毛一飞',GETDATE()-30)INSERT INTO T_Product_imgs (Pro_id,imgName,CreateUser,CreateDate) VALUES(1,'XL-20200801161136.jpg',N'DBA-JAVA','2020-08-01 23:20:56')
INSERT INTO T_Product_imgs (Pro_id,imgName,CreateUser,CreateDate) VALUES(2,'LT-20200906162236.jpg',N'DBA卢飞虎','2020-09-06 13:20:26')
INSERT INTO T_Product_imgs (Pro_id,imgName,CreateUser,CreateDate) VALUES(3,'DC-20210808163336.jpg',N'DBA卢飞虎','2021-08-08 23:20:56')
INSERT INTO T_Product_imgs (Pro_id,imgName,CreateUser,CreateDate) VALUES(3,'DC-20211005163336.jpg',N'DBA卢飞虎','2021-10-05 22:20:56')
INSERT INTO T_Product_imgs (Pro_id,imgName,CreateUser,CreateDate) VALUES(2,'LT-20210609162236.jpg',N'Oracle','2021-06-09 20:20:56')
INSERT INTO T_Product_imgs (Pro_id,imgName,CreateUser,CreateDate) VALUES(1,'XL-20210808161136.jpg',N'JAVA','2021-08-08 23:20:56')
INSERT INTO T_Product_imgs (Pro_id,imgName,CreateUser,CreateDate) VALUES(1,'XL-20200608161136.jpg',N'MongDB','2020-06-08 23:20:56')SELECT * FROM T_ProductSELECT * FROM T_Product_imgs ORDER BY Pro_id,CreateDate DESC 

在这里插入图片描述

  • 使用INNER JOIN 和子查询使用ROW_NUMBER() OVER字句
SELECT a.id,a.title,a.Tag,a.Price,T.Pro_id,T.imgName
FROM T_Product AS aINNER JOIN  ( SELECT Pro_id,imgName,CreateDate,ROW_NUMBER() OVER(PARTITION BY Pro_id  ORDER BY CreateDate DESC) AS Rn FROM T_Product_imgs AS b ) AS T
ON a.id=T.Pro_id
WHERE T.Rn=1

在这里插入图片描述

  • CROSS APPLY 实现
    与子查询的比较:与在 SELECT 语句中使用的子查询不同,CROSS APPLY 中的子查询或函数可以引用外部查询的列
SELECT a.id,a.title,a.Tag,a.Price,T.Pro_id,T.imgName
FROM T_Product as aCROSS APPLY ( SELECT TOP 1 b.pro_id,b.imgName FROM T_Product_imgs AS b WHERE b.Pro_id=a.id ORDER BY b.CreateDate DESC) AS T

在这里插入图片描述
在上面的示例中,对于 T_Product 表中的每一行,子查询都会返回最新的产品图片 imgName (如果存在)。如果某个产品没有产品图片,则该产品的信息不会出现在结果集中。

INSERT INTO T_Product (Title,Tag,Price,CreateUser,CreateDate) VALUES(N'大板X99',N'主板',230670.00,N'DBA毛一飞',GETDATE()-30)

在这里插入图片描述
插入T_Product 表的ID=4数据,但没有在T_Product_imgs表中插入最新的产品图片 imgName

SELECT a.id,a.title,a.Tag,a.Price,T.Pro_id,T.imgName
FROM T_Product as aCROSS APPLY ( SELECT TOP 1 b.pro_id,b.imgName FROM T_Product_imgs AS b WHERE b.Pro_id=a.id ORDER BY b.CreateDate DESC) AS T

再次执行,发现T_Product 表的ID=4数据是没有出现在结果集中。充分验证如果内部查询或函数对于外部查询的某行没有返回任何行,则该外部查询的行不会出现在结果集中。
在这里插入图片描述

这篇关于SQL Server中CROSS APPLY连接操作的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1071093

相关文章

Linux下MySQL数据库定时备份脚本与Crontab配置教学

《Linux下MySQL数据库定时备份脚本与Crontab配置教学》在生产环境中,数据库是核心资产之一,定期备份数据库可以有效防止意外数据丢失,本文将分享一份MySQL定时备份脚本,并讲解如何通过cr... 目录备份脚本详解脚本功能说明授权与可执行权限使用 Crontab 定时执行编辑 Crontab添加定

Java实现在Word文档中添加文本水印和图片水印的操作指南

《Java实现在Word文档中添加文本水印和图片水印的操作指南》在当今数字时代,文档的自动化处理与安全防护变得尤为重要,无论是为了保护版权、推广品牌,还是为了在文档中加入特定的标识,为Word文档添加... 目录引言Spire.Doc for Java:高效Word文档处理的利器代码实战:使用Java为Wo

MySQL中On duplicate key update的实现示例

《MySQL中Onduplicatekeyupdate的实现示例》ONDUPLICATEKEYUPDATE是一种MySQL的语法,它在插入新数据时,如果遇到唯一键冲突,则会执行更新操作,而不是抛... 目录1/ ON DUPLICATE KEY UPDATE的简介2/ ON DUPLICATE KEY UP

MySQL分库分表的实践示例

《MySQL分库分表的实践示例》MySQL分库分表适用于数据量大或并发压力高的场景,核心技术包括水平/垂直分片和分库,需应对分布式事务、跨库查询等挑战,通过中间件和解决方案实现,最佳实践为合理策略、备... 目录一、分库分表的触发条件1.1 数据量阈值1.2 并发压力二、分库分表的核心技术模块2.1 水平分

Python与MySQL实现数据库实时同步的详细步骤

《Python与MySQL实现数据库实时同步的详细步骤》在日常开发中,数据同步是一项常见的需求,本篇文章将使用Python和MySQL来实现数据库实时同步,我们将围绕数据变更捕获、数据处理和数据写入这... 目录前言摘要概述:数据同步方案1. 基本思路2. mysql Binlog 简介实现步骤与代码示例1

sysmain服务可以禁用吗? 电脑sysmain服务关闭后的影响与操作指南

《sysmain服务可以禁用吗?电脑sysmain服务关闭后的影响与操作指南》在Windows系统中,SysMain服务(原名Superfetch)作为一个旨在提升系统性能的关键组件,一直备受用户关... 在使用 Windows 系统时,有时候真有点像在「开盲盒」。全新安装系统后的「默认设置」,往往并不尽编

Python自动化处理PDF文档的操作完整指南

《Python自动化处理PDF文档的操作完整指南》在办公自动化中,PDF文档处理是一项常见需求,本文将介绍如何使用Python实现PDF文档的自动化处理,感兴趣的小伙伴可以跟随小编一起学习一下... 目录使用pymupdf读写PDF文件基本概念安装pymupdf提取文本内容提取图像添加水印使用pdfplum

Python 基于http.server模块实现简单http服务的代码举例

《Python基于http.server模块实现简单http服务的代码举例》Pythonhttp.server模块通过继承BaseHTTPRequestHandler处理HTTP请求,使用Threa... 目录测试环境代码实现相关介绍模块简介类及相关函数简介参考链接测试环境win11专业版python

Python从Word文档中提取图片并生成PPT的操作代码

《Python从Word文档中提取图片并生成PPT的操作代码》在日常办公场景中,我们经常需要从Word文档中提取图片,并将这些图片整理到PowerPoint幻灯片中,手动完成这一任务既耗时又容易出错,... 目录引言背景与需求解决方案概述代码解析代码核心逻辑说明总结引言在日常办公场景中,我们经常需要从 W

使用shardingsphere实现mysql数据库分片方式

《使用shardingsphere实现mysql数据库分片方式》本文介绍如何使用ShardingSphere-JDBC在SpringBoot中实现MySQL水平分库,涵盖分片策略、路由算法及零侵入配置... 目录一、ShardingSphere 简介1.1 对比1.2 核心概念1.3 Sharding-Sp