SqlServer 大数据量分页查询

SqlServer 分页查询,很多人习惯用 row_number 进行排序,在此排序的基础上利用 row_number 来处理分页。 但 row_number 这个函数有个缺点,表的行数越多时,大到十几吨行或几十万行时,row_number 实现的翻页,页码越大查询效果越慢,甚至直接查询超时。

下面给出几种翻页的实现方法 ,供大家测试和参考,废话不多说,直接上语句。

sql 复制代码
 SELECT COUNT(*) As Total FROM V_ME_mom_order_List  
 WHERE cOrgCode LIKE '01%'  AND  ( 1=1  OR ISNULL(cInvCode,'')='')  AND  ( 1=1  OR ISNULL(cInvCCode,'')='')  AND  ( 1=1  OR ISNULL(cDepCode,'')='')  

  --1. ROW_NUMBER  正向======== > 起始行 + TOP 页大小
  SELECT Top 500 1 FROM 
  (  
	  SELECT row_number()over(order by MOMID) as rowNum, 
	  1 as c
	  FROM V_ME_mom_order_List WITH(NOLOCK)  
	  WHERE cOrgCode LIKE '01%'  
		AND  ( 1=1  OR ISNULL(cInvCode,'')='')  
		AND  ( 1=1  OR ISNULL(cInvCCode,'')='')  
		AND  ( 1=1  OR ISNULL(cDepCode,'')='')  
  ) t 
  WHERE t.rowNum > 136500 order by t.rowNum asc

  --2. ROW_NUMBER 正向======== > 起始行 + <= 结束行
  SELECT 1 FROM (  
	  SELECT row_number()over(order by MOMID) as rowNum, 
	  1 as c 
	  FROM V_ME_mom_order_List WITH(NOLOCK)  
	  WHERE cOrgCode LIKE '01%'  
		AND  ( 1=1  OR ISNULL(cInvCode,'')='')  
		AND  ( 1=1  OR ISNULL(cInvCCode,'')='')  
		AND  ( 1=1  OR ISNULL(cDepCode,'')='')  
  ) t 
  WHERE t.rowNum > 136500 AND  t.rowNum <= 13700 order by t.rowNum asc
  
  --3. ROW_NUMBER 反向查询
  SELECT * FROM (  
	  SELECT row_number()over(order by MOMID desc) as rowNum, 
	  *
	  FROM V_ME_mom_order_List WITH(NOLOCK)  
	  WHERE cOrgCode LIKE '01%'  
		AND  ( 1=1  OR ISNULL(cInvCode,'')='')  
		AND  ( 1=1  OR ISNULL(cInvCCode,'')='')  
		AND  ( 1=1  OR ISNULL(cDepCode,'')='')  
  ) t 
  WHERE t.rowNum > 0 AND  t.rowNum <= 52
  order by t.rowNum asc


--4. OFFSET 正向
SELECT 1
FROM V_ME_mom_order_List
WHERE  cOrgCode LIKE '01%' 
	AND  ( 1=1  OR ISNULL(cInvCode,'')='')  
	AND  ( 1=1  OR ISNULL(cInvCCode,'')='')  
	AND  ( 1=1  OR ISNULL(cDepCode,'')='')    -- 如果有的话
ORDER BY MOMID ASC  -- 分页必须指定排序,否则结果不可靠
OFFSET (274 - 1) * 500 ROWS  -- 计算需要跳过的行数
FETCH NEXT 500 ROWS ONLY;  -- 获取当前页的100条数据




--5. OFFSET 反向
SELECT *
FROM V_ME_mom_order_List
WHERE  cOrgCode LIKE '01%' 
	AND  ( 1=1  OR ISNULL(cInvCode,'')='')  
	AND  ( 1=1  OR ISNULL(cInvCCode,'')='')  
	AND  ( 1=1  OR ISNULL(cDepCode,'')='')    -- 如果有的话
ORDER BY MOMID DESC  -- 分页必须指定排序,否则结果不可靠
OFFSET 0 ROWS  -- 计算需要跳过的行数
FETCH NEXT 52 ROWS ONLY;  -- 获取当前页的100条数据


 

上面几种方式 测试下来感觉 正向、offset 的效果最好。如果有 嵌套查询、子查询这些情况,子查询用 with 语句效果更好。

相关推荐
等....2 小时前
Minio使用
数据库
win x3 小时前
Redis 使用~如何在Java中连接使用redis
java·数据库·redis
迷枫7123 小时前
DM8 数据库安装实战:从零搭建达梦数据库环境(附全套工具链接)
数据库
XDHCOM4 小时前
PostgreSQL 25001: active_sql_transaction 报错原因分析,故障修复步骤详解,远程处理解决方案
数据库·sql·postgresql
卤炖阑尾炎4 小时前
PostgreSQL 日常运维全指南:从基础操作到备份恢复
运维·数据库·postgresql
daad7775 小时前
wifi_note
运维·服务器·数据库
xixingzhe26 小时前
Mysql统计空间增量
数据库·mysql
程序员萌萌7 小时前
Redis的缓存机制和淘汰策略详解
数据库·redis·缓存机制·淘汰策略
不剪发的Tony老师7 小时前
SQLite 3.53.0版本发布,重要更新
数据库·sqlite
Bczheng17 小时前
九.Berkeley DB数据库 序列化和钱包管理(1)
数据库