【SqlServer系列】子查询

1   概述

1.1  已发布【SqlServer系列】文章

1.2  本篇文章内容概要

子查询和表连接,主要是针对两张及以上表之间关联关系进行查询。子查询,按是否独立划分,分为独立子查询和相关子查询;按内部查询返回是否为单值,分为单值子查询(注意:应理解单值子查询和标量子查询关系)和多值子查询;如下表格内容为按子查询是否独立来划分的。

1.3 本章测试样表和sql

业务场景:

有两张表,分别为顾客表(顾客ID,顾客姓名,顾客公司),即Customers(CustID,CustName,CustCompany)和顾客订单表(顾客ID,订单ID,订单日期),即Orders(CustID,OrderID,OrderDate),两张表通过顾客ID(CustID)进行关联。

SQL语句分别如下:

(1)创建子查询DB:WJM_CHildQuery

1 --创建子查询数据库WJM_ChildQuery
2 IF DB_ID(‘WJM_CHildQuery‘) IS NOT NULL
3    DROP DATABASE ABC
4    GO
5    CREATE DATABASE WJM_ChildQuery

(2)创建Customers表并初始化

 1 USE WJM_CHildQuery
 2
 3 --CREATE TABLE Customers
 4 CREATE TABLE Customers
 5 (
 6    CustID INT IDENTITY(1,1)  NOT NULL PRIMARY KEY,  --顾客ID
 7    CustName  VARCHAR(50),--顾客姓名
 8    CustCompany VARCHAR(50) --顾客公司
 9 )
10
11 --Initial Customers
12
13 INSERT INTO Customers VALUES(‘赵武‘,‘A‘)
14 INSERT INTO Customers VALUES(‘刘杨‘,‘B‘)
15 INSERT INTO Customers VALUES(‘张永为‘,‘C‘)
16 INSERT INTO Customers VALUES(‘李龙飞‘,‘D‘)
17 INSERT INTO Customers VALUES(‘邓华‘,‘E‘)
18 INSERT INTO Customers VALUES(‘张涛明‘,‘F‘)

执行查询语句:

1 SELECT *
2 FROM Customers

查询结果为:

(2)创建Orders表并初始化

 1 USE WJM_CHildQuery
 2
 3 CREATE TABLE Orders
 4 (
 5   OrderID INT IDENTITY(1,1) PRIMARY KEY NOT NULL, --订单ID
 6   CustID INT NOT NULL,  --顾客ID
 7   OrderDate DATE --订单日期
 8 )
 9
10 --Initial Orders
11
12 INSERT INTO Orders VALUES(1,‘2015-06-25‘)
13 INSERT INTO Orders VALUES(2,‘2017-06-03‘)
14 INSERT INTO Orders VALUES(3,‘2016-04-25‘)
15 INSERT INTO Orders VALUES(4,‘2016-03-04‘)
16 INSERT INTO Orders VALUES(5,‘2013-03-04‘)
17 INSERT INTO Orders VALUES(6,‘2017-01-01‘)
18 INSERT INTO Orders VALUES(2,‘2016-08-25‘)
19 INSERT INTO Orders VALUES(5,‘2016-12-25‘)

执行查询语句:

1 SELECT *
2 FROM Orders

查询结果为:

2  独立子查询

2.1  独立标量子查询

问题:请用两种方法返回Orders表中的订单ID最大的订单信息。

方法一:标量单值子查询

1 --返回Orders表中订单ID最大的记录
2 SELECT OutQueryOrders.OrderID,OutQueryOrders.CustID,YEAR(OutQueryOrders.OrderDate) AS OrderDateYear
3 FROM Orders AS OutQueryOrders
4 WHERE OutQueryOrders.OrderID=
5 (
6    SELECT MAX(InnerQueryOrders.OrderID)
7    FROM Orders InnerQueryOrders
8 )

查询结果:

结果分析

a.期望结果

b.查询语句分析

方法二:采用变量

1 DECLARE @MaxOrderID AS INT=
2 (
3    SELECT MAX(OD.OrderID)
4    FROM Orders AS OD
5 )
6
7 SELECT *
8 FROM Orders
9 WHERE Orders.OrderID=@MaxOrderID

查询结果:

结果分析:

a.期望结果

b.查询语句分析

2.2  独立子查询

问题:返回订单表中订单ID>5的顾客信息

1 --返回订单表中订单ID>5的顾客信息
2 SELECT C.CustID,C.CustName,C.CustCompany
3 FROM Customers AS C
4 WHERE C.CustID  IN
5 (
6     SELECT Orders.CustID
7     FROM Orders
8     WHERE Orders.OrderID>5
9 )

查询结果:

结果分析:

a.期望结果

b.查询语句分析

2.3  小结

a.每个子查询均有所属的外部查询;
b.独立子查询独立于其外部查询;
c.可以把独立子查询单独运行;
d.独立子查询在执行外部查询之前先执行一次,接着外部查询再使用子查询的结果继续进行查询;
e.标量子查询只返回单个值,不管其是否是独立子查询;
f.标量子查询可以出现在外部查询中期望使用单个值的任何地方(WHERE、SELECT等)
g.算术运算符(>,>=,<,<=,!=,=)其后只能为单值;
h.如果标量子查询没有返回任何值,其结果就转换为NULL,和NULL进行比较得到的是UNKNOWN,查询过滤器不会返回任何让过滤表达式计算结果为UNKOWN的行;
i.多值子查询,一般与谓词连用,如IN,NOT IN,EXISTS等;
j.在子查询中,一般单值和多值是混合用,如常见与BETWEEN......AND.....连用等;

3   相关子查询

3.1  SQL示例及示例结果

问题:在订单表Orders表中,为每个客户返回其订单ID最大的订单

 1 --为每个客户返回其订单ID最大的订单
 2
 3 SELECT OutQueryOrders.OrderID,OutQueryOrders.CustID,YEAR(OutQueryOrders.OrderDate) AS OrderDateYear
 4 FROM Orders AS OutQueryOrders
 5 WHERE OutQueryOrders.OrderID=
 6 (
 7    SELECT MAX(InnerQueryOrders.OrderID)
 8    FROM Orders InnerQueryOrders
 9    WHERE OutQueryOrders.CustID=InnerQueryOrders.CustID
10 )

查询结果:

3.2  示例结果分析

a.期望结果

b.查询语句分析

3.3  小结

a.相关子查询内部查询依赖于外部查询,不能独立运行;
b.相关子查询内部查询中,受限制于外部查询的条件,如上述例子中的WHERE之后的条件;
c.在逻辑上,子查询会为每个外部查询单独计算一次;
d.多值子查询,一般与谓词连用,如IN,NOT IN,EXISTS等;
e.在子查询中,一般单值和多值是混合用,如常见与BETWEEN......AND.....连用等;

4   子查询拓展

关于子查询,还有很多内容,如返回前后记录查询,连续聚合,行为不当等,但这部分内容属于子查询的中高级部分,准备放在【SqlServer系列】中高级部分来与大家分享。

5   版权

  • 感谢您的阅读,若有不足之处,欢迎指教,共同学习、共同进步。
  • 博主网址:http://www.cnblogs.com/wangjiming/。
  • 极少部分文章利用读书、参考、引用、抄袭、复制和粘贴等多种方式整合而成的,大部分为原创。
  • 如您喜欢,麻烦推荐一下;如您有新想法,欢迎提出,邮箱:[email protected]。
  • 可以转载该博客,但必须著名博客来源。
时间: 2024-10-05 14:44:08

【SqlServer系列】子查询的相关文章

SQLServer(子查询,存储过程,透视,索引)

存储过程: 视图是对一个查询语句的封装,而存储过程:将一段t-sql脚本进行封装,对一段逻辑操作的封装. 创建存储过程: create proc 名称 参数表 as begin end 案例:去左右空格 creat proc trim @str1 varchar(10)asbegin select LTRIM(RTRIM(@str1)) end 在<可编程性>里面有<存储过程>里面有trim这个名 用的时候: exec trim '参数' 案例:编写存储过程,查询表中的总数居,以及

SQLServer —— EXISTS子查询

一.删除数据库 use master go if exists (select * from sysdatabases where name = 'Demo') drop database Demo go 二.删除表 use PhoneList go if exists (select * from sysobjects where name = 'PCategory') drop table PCategory go 三.子查询 有如下一张学员成绩表: 现在,有这么个需求,查询 C# 考试成绩

sqlserver根据子查询更新语句

语法结构: update 主表别名 set 主表别名.name=(select 子表别名.name from 子表 子表别名 where 主表别名.id=子表别名.id) from 主表 主表别名 例: update t set t.name=(select b.name from users b where t.userid=b.userid) from teacher t 原文地址:http://blog.51cto.com/1197822/2156632

sqlserver中的 数据转换 与 子查询

原文:sqlserver中的 数据转换 与 子查询 数据类型转换 --cast转换 select CAST(1.23 as int) select CAST(1.2345 as decimal(18,2)) select CAST(123 as varchar(10))     整型转换成字符串型 select CAST('123.333' as decimal(18,4))    字符串型转换成浮点型 --convert转换 select CONVERT(int,12.345) 子查询(嵌套查

SQL Server调优系列基础篇(子查询运算总结)

原文:SQL Server调优系列基础篇(子查询运算总结) 前言 前面我们的几篇文章介绍了一系列关于运算符的介绍,以及各个运算符的优化方式和技巧.其中涵盖:查看执行计划的方式.几种数据集常用的连接方式.联合运算符方式.并行运算符等一系列的我们常见的运算符.有兴趣的童鞋可以点击查看. 本篇我们介绍关于子查询语句的一系列内容,子查询一般是我们形成复杂查询的一些基础性操作,所以关于子查询的应用方式就非常重要. 废话少说,开始本篇的正题. 技术准备 数据库版本为SQL Server2008R2,利用微软

Sqlserver 使用CTE如何按子查询排序?

Sqlserver 使用CTE如何按子查询排序? 需求:查出最近有更改的客户信息(按最后更改时间排序,来自SystemLog表LogDateTime字段) 说明: Customer:客户信息表SystemLog:系统日志表,记录所有表信息的增,删,改 自己公司开发的行业软件,不仅自己公司有在用,其他公司也在用,本公司进行软件维护和提供升级服务由于之前的设计客户信息表并没有更改时间字段,而现在该系统已经在N个不同的数据库里运行导致不能随意更改数据库结构(虽然可以升级,但代价太大) 不能使用视图及存

Mysql高手系列 - 第12篇:子查询详解

这是Mysql系列第12篇. 环境:mysql5.7.25,cmd命令中进行演示. 本章节非常重要. 子查询 出现在select语句中的select语句,称为子查询或内查询. 外部的select查询语句,称为主查询或外查询. 子查询分类 按结果集的行列数不同分为4种 标量子查询(结果集只有一行一列) 列子查询(结果集只有一列多行) 行子查询(结果集有一行多列) 表子查询(结果集一般为多行多列) 按子查询出现在主查询中的不同位置分 select后面:仅仅支持标量子查询. from后面:支持表子查询

Oracle系列:(14)子查询

子查询的作用:查询条件未知的事物 查询条件已知的问题:例如:查询工资为800的员工信息 查询条件未知的问题:例如:查询工资为20号部门平均工资的员工信息 一个条件未知的问题,可以分解为多个条件已知的问题 查询工资比WARD高的员工信息 第一:查询WARD的工资?       select sal from emp where ename = 'WARD'; 第二:查询工资比1250高的员工信息?       select * from emp where sal > 1250; 子查询: sel

MySQL---数据库从入门走上大神系列(四)-子查询、表与表之间的关系

本篇博客讲解(子查询)非相关子查询/相关子查询,一对一,一对多,多对一,多对的的关系! 准备: 首先我们创建一列sex.再为部分行设置好值0(女)或者1(男): delete from stud where age=26; 删除年龄为26的行. update stud set sex='1' where saddress like '湖南%'; 将地址中湖南开头的人的的sex修改为1. update stud set sex='0' where age>=30 and sex is null;