T-SQL基础–TOP

T-SQL基础–TOP

 

理解TOP子句

 

众所周知,TOP子句可以通过控制返回行的数量来影响查询。

我们知道TOP子句能很容易的满足返回指定行数的子集,接下来有一些例子来展示什么情况下使用TOP子句来返回一个结果集;

  • 你打算返回的恰好是一个记录的子集来验证你代码;
  • 你仅仅需要确定至少一行数据满足特定的Where条件;
  • 你的业务需求指示你仅仅返回前面的几行数据,基于一个特定的Where条件;

为了去解释TOP子句的如何工作,我将列举几个实例,使你能够更容易理解并观察使用TOP子句的影响返回值得细微差别。

TOP 的语法

语法很简单,可以将TOP加在任何如 SELECT、DELETE, INSERT, or UPDATE 的语句中:

TOP (expression) [PERCENT]
[WITH TIES]

expression”的值是一个数字,,如果PERCENT的可选项被启用则数字将被转换成一个float 类型,否则姜维BIGINT类型。可以指定数字也可以使用局部变量。

可选项WITH TIES ,用来包含具有系统值得数据,需要注意的是该选项支队带有Order by的子句有效。

举例说明:

现有数据:

image

简单实例1

-- 找到两个SalesAgent 根据SalesAmount倒序。先排序在选出前两个,如果没有Order by 则此数据会随即返回(没有主键)
 
SELECT TOP(2) SalesAgent, Region, SalesAmount
FROM dbo.HectorSales
ORDER BY SalesAmount DESC;

运行结果:

SalesAgent                     Region     SalesAmount
------------------------------ ---------- --------------------------
Nick Potts                     East       9834212.87
Mary Johnson                   West       8723412.61

 

使用百分比的查询

If you want to return a percentage of the top records in a set then you need to use the TOP clause with the PERCENT option. To demonstrate using the PERCENT option look at Listing 3.

-- 查询前百分之50的数据,按照SalesAmount
SELECT TOP(50) PERCENT SalesAgent, Region, SalesAmount
FROM dbo.HectorSales
ORDER BY SalesAmount DESC;

查询结果:

SalesAgent                     Region     SalesAmount
----------------------------- ---------- ---------------------------
Nick Potts                     East       9834212.87
Mary Johnson                   West       8723412.61
Sam Holder                     East       8723412.61
Stan Morris                    East       4562341.67
Lori Morin                     East       2000111.67

使用变量的TOP查询

–不带百分比的查询

DECLARE @Number INT = 2;
SELECT TOP(@Number) SalesAgent, Region, SalesAmount
FROM dbo.HectorSales
ORDER BY SalesAmount DESC;

-- 带百分比的
SET @Number = 50;
SELECT TOP(@Number) PERCENT SalesAgent, Region, SalesAmount
FROM dbo.HectorSales
ORDER BY SalesAmount DESC;

使用 WITH TIES 可选项

按照值进行排序,如果有相同的则一并显示出来

-- 找出SalesAmount 最大的前两个。实际是3个
SELECT TOP(2) WITH TIES SalesAgent, Region, SalesAmount
FROM dbo.HectorSales
ORDER BY SalesAmount DESC;

查询结果:

SalesAgent                     Region     SalesAmount
------------------------------ ---------- --------------------------
Nick Potts                     East       9834212.87
Mary Johnson                   West       8723412.61
Sam Holder                     East       8723412.61

正如我们看到的,返回了3行而不是2行,因为第三行和第二行的值是相同的。

使用TOP子句实现更新

 

如何使用TOP子句限制更新的行数,如下:

UPDATE TOP (2) dbo.HectorSales
SET SalesAmount = 100000.00
FROM dbo.HectorSales
WHERE Region = ‘West’

更新后查询Region = 'West'的结果

SalesAgent                     Region     SalesAmount
------------------------------ ---------- --------------------------
John Smith                     West       100000.00
Mary Johnson                   West       100000.00
Doris Bean                     West       2000111.67
Martin Derrick                 West       120834.81
Don Olson                      West       508921.48

当然也可以使用其他不同的方式来更新,先查询出前2的数据,然后将符合前2的数据进行更新,如下:

UPDATE dbo.HectorSales 
SET SalesAmount = 6666666.00
FROM
(SELECT TOP(2) SalesAgent FROM dbo.HectorSales
WHERE Region = ‘West’
ORDER BY SalesAmount DESC) TS
WHERE dbo.HectorSales.SalesAgent = TS.SalesAgent

更新后的结果:

SalesAgent                     Region     SalesAmount
------------------------------ ---------- --------------------------
John Smith                     West       100000.00
Mary Johnson                   West       100000.00
Doris Bean                     West       6666666.00
Martin Derrick                 West       120834.81
Don Olson                      West       6666666.00

使用TOP完成Insert 语句

例如我打算将SalesAmount 最多的两个SalesAgent 插入到表dbo.TopTwoSales 的agent 里面。

INSERT TOP(2) INTO dbo.TopTwoHectorSales
SELECT * FROM dbo.HectorSales
ORDER BY SalesAmount DESC;
SELECT * FROM dbo.TopTwoHectorSales;

结果集如下:

SalesAgent                     Region     SalesAmount
------------------------------ ---------- --------------------------
John Smith                     West       100000.00
Mary Johnson                   West       100000.00

通过查询结果我们发现插入的两行,并不是SalesAmount 最大的两行,因为我将TOP放在了Insert 后面,SQLServer 认为从子结果集中的前两行,这样的话实际上子结果集是随即的。为了纠正之前的问题,我这样写:

INSERT INTO dbo.TopTwoHectorSales
SELECT TOP(2) * FROM dbo.HectorSales
ORDER BY SalesAmount DESC;
SELECT * FROM dbo.TopTwoHectorSales;

改正后的结果:

SalesAgent                     Region     SalesAmount
------------------------------ ---------- --------------------------
Nick Potts                     East       9834212.87
Sam Holder                     East       8723412.61

使用TOP完成DELETE语句

BEGIN TRANSACTION;
DELETE FROM dbo.HectorSales
WHERE SalesAgent in (SELECT TOP(2) SalesAgent FROM dbo.HectorSales
                     ORDER by SalesAmount ASC);
SELECT * FROM dbo.HectorSales;
ROLLBACK;

结果如下:

SalesAgent                     Region     SalesAmount
------------------------------ ---------- --------------------------
John Smith                     West       100000.00
Doris Bean                     West       6666666.00
Martin Derrick                 West       120834.81
Don Olson                      West       6666666.00
Sam Holder                     East       8723412.61
Nick Potts                     East       9834212.87
Lori Morin                     East       2000111.67
Stan Morris                    East       4562341.67

Report 12: Rows inserted into dbo.TopTwoHectorSales table using ORDER BY

当然如果想包含相同的值,使用WITH TIES

DELETE FROM dbo.HectorSales
WHERE SalesAgent in (SELECT TOP(2) WITH TIES SalesAgent FROM dbo.HectorSales
                     ORDER by SalesAmount ASC);
SELECT * FROM dbo.HectorSales;

与上一个相比,这个代码将会多删除掉一个SalesAmount 1000000.00数据

谨慎使用TOP关键在UNION、EXCEPT和INTERSECT语句中

创建一个表,插入初始数据。

CREATE TABLE dbo.Sales (
AgentName varchar(30),
Region varchar(10),
SalesAmount decimal(10,2));
INSERT INTO dbo.Sales VALUES
    (‘John Smith’, ‘West’,1012302.01),
    (‘Mary Johnson’, ‘West’,2453202.89),
    (‘Doris Bean’, ‘West’,99001.43),
    (‘Sam Holder’, ‘East’,8723412.61),
    (‘Nick Potts’, ‘East’,9834212.87),
    (‘Jason Thomas’, ‘East’,13424.51);

将东西部最高的SalesAmount 筛选出来并联

SELECT AgentName, Region, SalesAmount FROM
(
SELECT TOP(1) AgentName, Region, SalesAmount 
	FROM dbo.Sales
	WHERE Region = 'East'
	ORDER BY SalesAmount DESC ) East
UNION  
SELECT AgentName, Region, SalesAmount FROM
(
SELECT TOP(1) AgentName, Region, SalesAmount
	FROM dbo.Sales
	WHERE Region = 'West'
	ORDER BY SalesAmount DESC ) West
ORDER BY SalesAmount DESC;

总结

 

TOP关键字,让我们可以返回指定行数的数据,也能指定行数或者百分比的数据。为了确保结果集的一致性,一定要保证使用ORDER BY,假如你没有使用则将返回随机的指定行数数据。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/119732.html原文链接:https://javaforall.net

(0)
上一篇 2021年11月25日 下午1:00
下一篇 2021年11月26日 上午6:00


相关推荐

  • MAC如何查看Tensorflow版本号[通俗易懂]

    MAC如何查看Tensorflow版本号[通俗易懂]详细教程:MAC如何查看Tensorflow版本号#首先打开MAC终端(terminal)1、激活tensorflow;2、然后进入python(根据版本不同输入自带版本号)3、输入python语句执行查询PS:必须在激活tensorflow环境后再输入python命令,否则会识别不到tensorflow,激活后可以看到在使用python前后命令前面都是有——(tensorflow)。示例代码如下:e@MacBookPro~%cd~/tensorflowe@MacBookProte

    2022年6月25日
    70
  • java转义字符之换行字符

    java转义字符之换行字符我们知道,java代码中,\r和\n和\r\n都表示换行,当我们想输出“\n”时,如果直接写System.println(“\n”);是看不到任何东西的,因为其自动被转义为换行了。这时,我们就需要使用转义字符了。这里先给出转义字符:使用示例:输出结果为:注: \r是回车符;&nbsp…

    2022年7月7日
    126
  • 四大审计费收费标准_四大 税务咨询收费

    四大审计费收费标准_四大 税务咨询收费“7年的心血和积累,说关就要关,说停就要停。没有人会甘心,但也早料到这一刻会突然到来。现在所能做的,唯有面对现实,准备好勇气,即使全部推倒从头再来,我们也绝不放弃。”1月22日,国内著名资源分享网站VeryCD的创始人黄一孟的这样一条微博牵动了不少用户的神经。用户们发现,VeryCD网站上的音乐频道已经悄然关闭,电影和电视剧频道也不再提供下载链接。坊间则风传VeryCD未取得广电总局的视听服务许

    2022年8月10日
    17
  • 投巧解决JavaScript split方法出现空字符的问题

    投巧解决JavaScript split方法出现空字符的问题直接使用split,前后各有一个“”值。>>varstr=’,a,b,c,d,e,f,’;>>str.split(‘,’);//(8)[“”,”a”,”b”,”c”,”d”,”e”,”f”,””]临时方法:split后,可以用filter过滤掉空值。>>varstr=’,a,b,c,d,e,f,’;>&…

    2025年8月8日
    6
  • sql格式化工具-SQL Pretty Printer

    sql格式化工具-SQL Pretty Printersql格式化工具-SQLPrettyPrinter导言使用主要功能:其他功能:RenameTable和RenameColumnToC#(ToVB.NET)C#ToSQL(VB.NETToSQL)SQLFormatterOptions导言写SQL语句或者脚本时,很容易出现凌乱的情况,这里推荐一款很方便的SQLServer插件:SQLPrettyPrinter…

    2025年8月21日
    6
  • fstream的用法_fun 的用法

    fstream的用法_fun 的用法在C++中输入输出到指定文件,或者从指定文件中读出数据使用fstream类较为方便。C++中处理文件类似于处理标准输入和标准输出。类ifstream、ofstream(向文件中写入)和fstream分别从类istream、ostream和iostream派生而来。作为派生的类,它们继承了插入和提取运算符(以及其他成员函数),还有与文件一起使用的成员和构造函数。可将文件<fstream&gt…

    2026年1月23日
    3

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号