[转]SQL Server 存储过程的分页方案比拼

本文涉及的产品
RDS SQL Server Serverless,2-4RCU 50GB 3个月
推荐场景:
云数据库 RDS SQL Server,基础系列 2核4GB
简介:   1SQL Server 存储过程的分页,这个问题已经讨论过几年了,很多朋友在问我,所以在此发表一下我的观点   2建立表:   3  4CREATE TABLE [TestTable] (   5 [ID] [int] IDENTITY (1, 1) NOT NULL ,   6...
 
  1 img_a6339ee3e57d1d52bc7d02b338e15a60.gif SQL Server 存储过程的分页,这个问题已经讨论过几年了,很多朋友在问我,所以在此发表一下我的观点
  2 img_a6339ee3e57d1d52bc7d02b338e15a60.gif建立表:
  3 img_a6339ee3e57d1d52bc7d02b338e15a60.gif
  4 img_a6339ee3e57d1d52bc7d02b338e15a60.gif CREATE   TABLE   [ TestTable ]  (
  5 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  [ ID ]   [ int ]   IDENTITY  ( 1 1 NOT   NULL  ,
  6 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  [ FirstName ]   [ nvarchar ]  ( 100 ) COLLATE Chinese_PRC_CI_AS  NULL  ,
  7 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  [ LastName ]   [ nvarchar ]  ( 100 ) COLLATE Chinese_PRC_CI_AS  NULL  ,
  8 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  [ Country ]   [ nvarchar ]  ( 50 ) COLLATE Chinese_PRC_CI_AS  NULL  ,
  9 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  [ Note ]   [ nvarchar ]  ( 2000 ) COLLATE Chinese_PRC_CI_AS  NULL  
 10 img_a6339ee3e57d1d52bc7d02b338e15a60.gif ON   [ PRIMARY ]
 11 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  GO
 12 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 13 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  
 14 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 插入数据:(2万条,用更多的数据测试会明显一些)
 15 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  SET   IDENTITY_INSERT  TestTable  ON
 16 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 17 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  declare   @i   int
 18 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  set   @i = 1
 19 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  while   @i <= 20000
 20 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  begin
 21 img_a6339ee3e57d1d52bc7d02b338e15a60.gif      insert   into  TestTable( [ id ] , FirstName, LastName, Country,Note)  values ( @i ' FirstName_XXX ' , ' LastName_XXX ' , ' Country_XXX ' , ' Note_XXX ' )
 22 img_a6339ee3e57d1d52bc7d02b338e15a60.gif      set   @i = @i + 1
 23 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  end
 24 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 25 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  SET   IDENTITY_INSERT  TestTable  OFF
 26 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 27 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  
 28 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 29 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  -- -----------------------------------
 30 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  
 31 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 分页方案一:(利用Not In和SELECT TOP分页)
 32 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 语句形式:
 33 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  SELECT   TOP   10   *
 34 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  FROM  TestTable
 35 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  WHERE  (ID  NOT   IN
 36 img_a6339ee3e57d1d52bc7d02b338e15a60.gif           ( SELECT   TOP   20  id
 37 img_a6339ee3e57d1d52bc7d02b338e15a60.gif           FROM  TestTable
 38 img_a6339ee3e57d1d52bc7d02b338e15a60.gif           ORDER   BY  id))
 39 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  ORDER   BY  ID
 40 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 41 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 42 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  SELECT   TOP  页大小  *
 43 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  FROM  TestTable
 44 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  WHERE  (ID  NOT   IN
 45 img_a6339ee3e57d1d52bc7d02b338e15a60.gif           ( SELECT   TOP  页大小 * 页数 id
 46 img_a6339ee3e57d1d52bc7d02b338e15a60.gif           FROM  表
 47 img_a6339ee3e57d1d52bc7d02b338e15a60.gif           ORDER   BY  id))
 48 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  ORDER   BY  ID
 49 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 50 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  -- -----------------------------------
 51 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  
 52 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 分页方案二:(利用ID大于多少和SELECT TOP分页)
 53 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 语句形式:
 54 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  SELECT   TOP   10   *
 55 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  FROM  TestTable
 56 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  WHERE  (ID  >
 57 img_a6339ee3e57d1d52bc7d02b338e15a60.gif           ( SELECT   MAX (id)
 58 img_a6339ee3e57d1d52bc7d02b338e15a60.gif           FROM  ( SELECT   TOP   20  id
 59 img_a6339ee3e57d1d52bc7d02b338e15a60.gif                   FROM  TestTable
 60 img_a6339ee3e57d1d52bc7d02b338e15a60.gif                   ORDER   BY  id)  AS  T))
 61 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  ORDER   BY  ID
 62 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 63 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 64 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  SELECT   TOP  页大小  *
 65 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  FROM  TestTable
 66 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  WHERE  (ID  >
 67 img_a6339ee3e57d1d52bc7d02b338e15a60.gif           ( SELECT   MAX (id)
 68 img_a6339ee3e57d1d52bc7d02b338e15a60.gif           FROM  ( SELECT   TOP  页大小 * 页数 id
 69 img_a6339ee3e57d1d52bc7d02b338e15a60.gif                   FROM  表
 70 img_a6339ee3e57d1d52bc7d02b338e15a60.gif                   ORDER   BY  id)  AS  T))
 71 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  ORDER   BY  ID
 72 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 73 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 74 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  -- -----------------------------------
 75 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  
 76 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 分页方案三:(利用SQL的游标存储过程分页)
 77 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  create    procedure  XiaoZhengGe
 78 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  @sqlstr   nvarchar ( 4000 ),  -- 查询字符串
 79 img_a6339ee3e57d1d52bc7d02b338e15a60.gif   @currentpage   int -- 第N页
 80 img_a6339ee3e57d1d52bc7d02b338e15a60.gif   @pagesize   int   -- 每页行数
 81 img_a6339ee3e57d1d52bc7d02b338e15a60.gif   as
 82 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  set  nocount  on
 83 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  declare   @P1   int -- P1是游标的id
 84 img_a6339ee3e57d1d52bc7d02b338e15a60.gif    @rowcount   int
 85 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  exec  sp_cursoropen  @P1  output, @sqlstr , @scrollopt = 1 , @ccopt = 1 , @rowcount = @rowcount  output
 86 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  select   ceiling ( 1.0 * @rowcount / @pagesize as  总页数 -- ,@rowcount as 总行数,@currentpage as 当前页 
 87 img_a6339ee3e57d1d52bc7d02b338e15a60.gif   set   @currentpage = ( @currentpage - 1 ) * @pagesize + 1
 88 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  exec  sp_cursorfetch  @P1 , 16 , @currentpage , @pagesize  
 89 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  exec  sp_cursorclose  @P1
 90 img_a6339ee3e57d1d52bc7d02b338e15a60.gif  set  nocount  off
 91 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 92 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 其它的方案:如果没有主键,可以用临时表,也可以用方案三做,但是效率会低。
 93 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 建议优化的时候,加上主键和索引,查询效率会提高。
 94 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 
 95 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 通过SQL 查询分析器,显示比较:我的结论是:
 96 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 分页方案二:(利用ID大于多少和SELECT TOP分页)效率最高,需要拼接SQL语句
 97 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 分页方案一:(利用Not In和SELECT TOP分页)   效率次之,需要拼接SQL语句
 98 img_a6339ee3e57d1d52bc7d02b338e15a60.gif 分页方案三:(利用SQL的游标存储过程分页)    效率最差,但是最为通用
 99 img_a6339ee3e57d1d52bc7d02b338e15a60.gif
100 img_a6339ee3e57d1d52bc7d02b338e15a60.gif在实际情况中,要具体分析。
相关实践学习
使用SQL语句管理索引
本次实验主要介绍如何在RDS-SQLServer数据库中,使用SQL语句管理索引。
SQL Server on Linux入门教程
SQL Server数据库一直只提供Windows下的版本。2016年微软宣布推出可运行在Linux系统下的SQL Server数据库,该版本目前还是早期预览版本。本课程主要介绍SQLServer On Linux的基本知识。 相关的阿里云产品:云数据库RDS&nbsp;SQL Server版 RDS SQL Server不仅拥有高可用架构和任意时间点的数据恢复功能,强力支撑各种企业应用,同时也包含了微软的License费用,减少额外支出。 了解产品详情:&nbsp;https://www.aliyun.com/product/rds/sqlserver
目录
相关文章
|
1月前
|
SQL 关系型数据库 MySQL
SQL中如何实现分页?
【8月更文挑战第3天】SQL中如何实现分页?
73 36
|
9天前
|
JSON 数据格式 Java
化繁为简的魔法:Struts 2 与 JSON 联手打造超流畅数据交换体验,让应用飞起来!
【8月更文挑战第31天】在现代 Web 开发中,JSON 成为数据交换的主流格式,以其轻量、易读和易解析的特点受到青睐。Struts 2 内置对 JSON 的支持,结合 Jackson 库可便捷实现数据传输。本文通过具体示例展示了如何在 Struts 2 中进行 JSON 数据的序列化与反序列化,并结合 AJAX 技术提升 Web 应用的响应速度和用户体验。
21 0
|
9天前
|
存储 SQL 数据库
如何使用 SQL Server 创建存储过程?
【8月更文挑战第31天】
42 0
|
2月前
|
SQL 存储 监控
SQL Server的并行实施如何优化?
【7月更文挑战第23天】SQL Server的并行实施如何优化?
53 13
|
2月前
|
SQL
解锁 SQL Server 2022的时间序列数据功能
【7月更文挑战第14天】要解锁SQL Server 2022的时间序列数据功能,可使用`generate_series`函数生成整数序列,例如:`SELECT value FROM generate_series(1, 10)。此外,`date_bucket`函数能按指定间隔(如周)对日期时间值分组,这些工具结合窗口函数和其他时间日期函数,能高效处理和分析时间序列数据。更多信息请参考官方文档和技术资料。
|
2月前
|
SQL 存储 网络安全
关系数据库SQLserver 安装 SQL Server
【7月更文挑战第26天】
43 6
|
27天前
|
SQL 安全 Java
驱动程序无法通过使用安全套接字层(SSL)加密与 SQL Server 建立安全连接。错误:“The server selected protocol version TLS10 is not accepted by client
驱动程序无法通过使用安全套接字层(SSL)加密与 SQL Server 建立安全连接。错误:“The server selected protocol version TLS10 is not accepted by client
98 0
|
2月前
|
存储 SQL C++
对比 SQL Server中的VARCHAR(max) 与VARCHAR(n) 数据类型
【7月更文挑战7天】SQL Server 中的 VARCHAR(max) vs VARCHAR(n): - VARCHAR(n) 存储最多 n 个字符(1-8000),适合短文本。 - VARCHAR(max) 可存储约 21 亿个字符,适合大量文本。 - VARCHAR(n) 在处理小数据时性能更好,空间固定。 - VARCHAR(max) 对于大文本更合适,但可能影响性能。 - 选择取决于数据长度预期和业务需求。
|
2月前
|
SQL Oracle 关系型数据库
MySQL、SQL Server和Oracle数据库安装部署教程
数据库的安装部署教程因不同的数据库管理系统(DBMS)而异,以下将以MySQL、SQL Server和Oracle为例,分别概述其安装部署的基本步骤。请注意,由于软件版本和操作系统的不同,具体步骤可能会有所变化。
113 3
|
2月前
|
SQL 存储 安全
数据库数据恢复—SQL Server数据库出现逻辑错误的数据恢复案例
SQL Server数据库数据恢复环境: 某品牌服务器存储中有两组raid5磁盘阵列。操作系统层面跑着SQL Server数据库,SQL Server数据库存放在D盘分区中。 SQL Server数据库故障: 存放SQL Server数据库的D盘分区容量不足,管理员在E盘中生成了一个.ndf的文件并且将数据库路径指向E盘继续使用。数据库继续运行一段时间后出现故障并报错,连接失效,SqlServer数据库无法附加查询。管理员多次尝试恢复数据库数据但是没有成功。
下一篇
DDNS