前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【愚公系列】2022年01月 SQL Server数据库-数据分页的五种性能分析

【愚公系列】2022年01月 SQL Server数据库-数据分页的五种性能分析

作者头像
愚公搬代码
发布2023-01-06 09:07:46
7260
发布2023-01-06 09:07:46
举报
文章被收录于专栏:历史专栏历史专栏

文章目录


前言

数据分页往往有三种常用方案。

  • 把数据库中存放的相关数据,全部通过编程语言读入内存中,再由代码对其进行分页操作(速度慢,简易性高)。
  • 直接在数据库中对相关数据进行分页操作,再把分页后的数据输出给代码程序(速度中,简易性中)。
  • 先把数据库中的相关数据全部读入“缓存”或第三方工具,再由代码程序对“缓存”或第三方工具中的数据进行读取+分页操作(速度快,简易性差)。

本文主要是直接在数据库中对相关数据进行分页操作,数据库是SQL Server上的案例(其它种类数据库由于Sql语句略有差异,所以需要调整,但方案也类似)

一、数据分页的五种性能分析

1.ROW_NUMBER() OVER()方式

1、这种分页方案主要是在SQL2012以下推荐使用。

通用写法如下:

代码语言:javascript
复制
--pageIndex 表示指定页
--pageSize  表示每页显示的条数
SELECT * FROM
    (SELECT ROW_NUMBER() OVER(ORDER BY 排序字段) AS RowId,* FROM 表名 ) AS r 
WHERE  RowId  BETWEEN ((pageIndex-1)*pageSize + 1) AND (pageIndex * PageSize)

用子查询新增一列行号(ROW_NUMBER)RowId查询,比较高效的查询方式,只有在SQL Server2005或更高版本才支持。

BETWEEN 1 AND 10 是指查询第1到第10条数据(闭区间),在这里面需要注意的是OVER的括号里面可以写多个排序字段。

2、代码案例

代码语言:javascript
复制
-- 1.数据库分页方案一  ROW_NUMBER() OVER()方式
SELECT * FROM
    (SELECT ROW_NUMBER() OVER(ORDER BY MO_ID) AS RowId,* FROM MO ) AS r 
WHERE  RowId BETWEEN 1 AND 10
在这里插入图片描述
在这里插入图片描述

2.offset fetch next方式

1、这种分页方案主要是在SQL2012及以上的版本才支持:推荐使用

通用写法如下:

代码语言:javascript
复制
--pageIndex 表示指定页
--pageSize  表示每页显示的条数
SELECT * FROM 表名 
ORDER BY 排序字段 offset ((pageIndex - 1) * pageSize) ROWS FETCH NEXT pageSize ROWS ONLY
  • offset 是跳过多少行,
  • next是取接下来的多少行,

句式 offset…rows fetch nect …rows only ,注意rows和末尾的only 不要写漏掉了,并且这种方式必须要接着Order by XX 使用,不然会报错。

2、代码案例

代码语言:javascript
复制
-- 2.数据库分页方案一  ROW_NUMBER() OVER()方式
SELECT * FROM MO 
ORDER BY MO_ID offset 0 ROWS FETCH NEXT 10 ROWS ONLY
在这里插入图片描述
在这里插入图片描述

3.top not in方式

1、不推荐使用这种方式进行分页

通用写法如下:

代码语言:javascript
复制
--pageIndex 表示指定页
--pageSize  表示每页显示的条数
SELECT TOP pageSize *
FROM 表名
WHERE 主键字段 NOT IN (SELECT TOP ((pageSize-1)*pageIndex) 主键字段 FROM 表名)

这条语句的原理是先查询1-10条记录的ID,然后再查询ID不属于这1-10条记录的ID,并且只需要10条记录,因为每页大小就是10,这就是获取到的第11-20条记录,这是非常简单的一种写法。

另外IN语句与NOT IN语句类似,这是NOT IN的写法,但是这种写法数据量大的话效率太低。

2、代码案例

代码语言:javascript
复制
-- 2.数据库分页方案一  ROW_NUMBER() OVER()方式
SELECT TOP 10 *
FROM MO 
WHERE MO_ID NOT IN (SELECT TOP 10 MO_ID FROM MO)
在这里插入图片描述
在这里插入图片描述

4.升序与降序方式

1、不推荐使用这种方式进行分页

通用写法如下:

代码语言:javascript
复制
--pageIndex 表示指定页
--pageSize  表示每页显示的条数
SELECT * FROM(
    SELECT TOP pageSize * FROM(
        SELECT TOP ((pageIndex - 1) * pageSize +(pageSize*2)) * FROM 表名 ORDER BY 排序字段 ASC) 
            AS TEMP1 ORDER BY 排序字段 DESC)
        AS TEMP2 ORDER BY 排序字段 ASC

这条语句首先查询前20条记录,然后在倒序查询前10条记录(即倒数10条记录),这个时候就已经获取到了11-20条记录,但是他们的顺序是倒序,所以最后又进行升序排序。

2、代码案例

代码语言:javascript
复制
--4.查询第11-20条记录
SELECT * FROM(
    SELECT TOP 10 * FROM(
        SELECT TOP 20 * FROM MO ORDER BY MO_ID ASC) 
            AS TEMP1 ORDER BY MO_ID DESC)
        AS TEMP2 ORDER BY MO_ID ASC
在这里插入图片描述
在这里插入图片描述

5.采用MAX(ID)或者MIN(ID)函数方式

1、不推荐使用这种方式进行分页

通用写法如下:

代码语言:javascript
复制
--pageIndex 表示指定页
--pageSize  表示每页显示的条数
SELECT TOP pageSize * FROM 表名 WHERE 排序字段>
    (SELECT MAX(menuId) FROM(SELECT TOP ((PageIndex-1)*PageSize) 排序字段 FROM 表名 ORDER BY 排序字段) AS TEMP1) --(第10条的id)

这个理解起来也简单,先把第10条记录的id找出来(当然这里面是直接使用MAX()进行查找,MIN()函数的用法也是类似的),然后再对比取比第10条记录的id大的前10条记录即为我们需要的结果。

2、代码案例

代码语言:javascript
复制
-- 5.查询第11-20条记录
SELECT TOP 10 * FROM MO WHERE MO_ID>
    (SELECT MAX(MO_ID) FROM(SELECT TOP 10 MO_ID FROM MO ORDER BY MO_ID) AS TEMP1) --(第10条的id)
在这里插入图片描述
在这里插入图片描述
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2023-01-05,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 文章目录
  • 前言
  • 一、数据分页的五种性能分析
    • 1.ROW_NUMBER() OVER()方式
      • 2.offset fetch next方式
        • 3.top not in方式
          • 4.升序与降序方式
            • 5.采用MAX(ID)或者MIN(ID)函数方式
            相关产品与服务
            数据库
            云数据库为企业提供了完善的关系型数据库、非关系型数据库、分析型数据库和数据库生态工具。您可以通过产品选择和组合搭建,轻松实现高可靠、高可用性、高性能等数据库需求。云数据库服务也可大幅减少您的运维工作量,更专注于业务发展,让企业一站式享受数据上云及分布式架构的技术红利!
            领券
            问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档