Oracle之分页查询

分页查询

在Oracle中实现分页的方法大致分为两种,用ROWNUM关键字和用ROWID关键字

1.ROWNUM

SELECT *
  FROM (SELECT ROW_.*, ROWNUM ROWNUM_
          FROM (SELECT *
                  FROM TABLE_Name
                 WHERE Name_ID = xxx
                 ORDER BY Age DESC) ROW_
         WHERE ROWNUM <= 20)
 WHERE ROWNUM_ >= 15;

这应该是我们大部分程序里所用到的版本,因为这个版本很容易实现复用,中间ROW_部分,就是我们平常写到的sql语句,然后再将起始条数和终止条数作为专门的分页sql语句传入即可查询出我们想要的结果。

从效率上看,上面的SQL语句在大多数情况拥有较高的效率,主要体现在WHERE ROWNUM <= 20这句上,这样就控制了查询过程中的最大记录数,而在查询的最外层控制最小值。但最大值意味着如果查到了很大的范围(如百万级别的数据),查询就会从很大范围内往里减少,效率就会很低,因此,当面对大数据量时或者优化查询效率时,如果你用了ROWNUM,可以换第二种方法

由以上的方法,又可以引申出3种方式:

(1)BETWEEN AND 

SELECT *
  FROM (SELECT A.*, ROWNUM RN
          FROM (SELECT *
                  FROM TABLE
                 WHERE Name_ID = xxx
                 ORDER BY Age DESC) A)
 WHERE RN BETWEEN 15 AND 20;

这个查询效率更低,因为:

Oracle可以将外层的查询条件推到内层查询中,以提高内层查询的执行效率,但不能跨越多层

由于查询条件BETWEEN 10 AND 20是存在于查询的第三层,而Oracle无法将第三层的查询条件推到最内层(即使推到最内层也没有意义,因为最内层查询不知道RN代表什么)。因此,这个查询语句,Oracle最内层返回给中间层的是所有满足条件的数据,而中间层返回给最外层的也是所有数据。数据的过滤在最外层完成,显然这个效率要比原始的查询低得多。

(2)MINUS

SELECT *
  FROM TABLE1
 WHERE ROWNUM <= 20
MINUS
SELECT * FROM TABLE1 WHERE ROWNUM <= 10;

查询了两次,效率上更差了一些

(3)ROW_NUMBER() OVER( ORDER BY ORDER_DATE DESC) 

这个和ROWNUM关键字类似,生成的顺序和rownum的语句一样,效率也一样(对于同样有ORDER BY 的ROWNUM语句来说),所以在这种情况下两种用法是一样的。 

而对于分组后查询做分页的话,则是ROWNUM无法实现的,这时只有ROW_NUMBER可以实现,ROW_NUMBER() OVER(PARTITION BY 分组字段 ORDER BY 排序字段)就能实现分组后编号,其代码为:

SELECT *
  FROM (SELECT a.*,
               ROW_NUMBER() OVER(PARTITION BY TRUNC(order_date) ORDER BY order_date DESC) rn
          FROM TABLE1 a)
 WHERE rn <= 10;

2.ROWID

ROWID仍旧需求ROWNUM,但方式不同,因此将其归为另一大类,其代码为:

SELECT *
  FROM (SELECT RID
          FROM (SELECT R.RID, ROWNUM LINENUM
                  FROM (SELECT ROWID RID
                          FROM TABLE1
                         WHERE TABLE1_ID = XX
                         ORDER BY order_date DESC) R
                 WHERE ROWNUM <= 20)
         WHERE LINENUM >= 10) T1,
       TABLE1 T2
 WHERE T1.RID = T2.ROWID;

从语句上看,共有4层Select嵌套查询,最内层为可替换的不分页原始SQL语句,但是他查询的字段只有ROWID,而没有任何待查询的实际表字段,具体查询实际字段值是在最外层实现的; 
这种方式的原理大致为:

    首先通过ROWNUM查询到分页之后的10条实际返回记录的ROWID,最后通过ROWID将最终返回字段值查询出来并返回

和前面ROWNUM实现方式相比,该SQL的实现方式更加繁琐,通用性也不是非常好,因为要将原始的查询语句分成两部分(查询字段在最外层,表及其查询条件在最内层),想要复用就很困难了; 但这种实现在特定场景下还是有优势的:比如我们经常要翻页到很后面,比如10000条记录中我们经常需要查9000-9100及其以后的数据;此时该方案效率可能要比前面的高; 因为前面的方案中是通过ROWNUM <= 9100来控制的,这样就需要查询出9100条数据,然后取最后9000-9100之间的数据,而这个方案直接通过ROWID取需要的那100条数据; 从不断向后翻页这个角度来看,第一种实现方案的成本会越来越高,基本上是线性增长,而第三种方案的成本则不会像前者那样快速,他的增长只体现在通过查询条件读取ROWID的部分;

因此,在我们实际项目中,基本分页都是可以单靠ROWNUM就可以实现,而在数据量只有几十万的情况下,效率也是够的,如果一定要优化,则可以考虑ROWID

  • 2
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Mybatis实现分页查询可以通过使用分页插件来实现。首先,在Mybatis配置文件中配置分页插件,然后在Mapper.xml文件中编写查询语句,通过传递参数指定查询起始位置和每页查询的数量即可实现分页查询。具体实现可参考Mybatis官方文档。 ### 回答2: Mybatis是一个基于Java的持久层框架,它可以方便地实现分页查询功能。下面我将介绍一下如何使用Mybatis实现分页查询。 首先,在Mybatis的配置文件中配置分页插件。在 `<configuration> </configuration>` 标签中添加以下代码: ```xml <plugins> <plugin interceptor="com.github.pagehelper.PageInterceptor"> <property name="dialect" value="mysql"/> <!-- 数据库方言,可以根据自己的实际情况进行配置 --> </plugin> </plugins> ``` 然后,在需要进行分页查询的Mapper接口中定义方法。在方法的参数列表中添加 `@Param` 注解,并添加Page类型的参数,用于传递分页参数。例如: ```java List<User> queryUserByPage(@Param("page") Page page); ``` 接着,在Mapper的XML映射文件中编写SQL语句实现分页查询。例如: ```xml <select id="queryUserByPage" resultType="com.example.User"> SELECT * FROM user LIMIT #{page.startRow}, #{page.pageSize} </select> ``` 最后,在Service层调用Mapper接口的分页查询方法,并通过PageHelper工具类设置分页参数。例如: ```java public PageInfo<User> queryUserByPage(int pageNum, int pageSize) { // 设置分页参数 PageHelper.startPage(pageNum, pageSize); // 调用Mapper接口的方法进行分页查询 List<User> userList = userMapper.queryUserByPage(); // 使用PageInfo类对查询结果进行封装 PageInfo<User> pageInfo = new PageInfo<>(userList); return pageInfo; } ``` 通过以上步骤,我们就可以使用Mybatis实现分页查询了。在调用Service层的分页查询方法时,只需要传入当前页码和每页显示的数据条数即可。Mybatis会自动根据传入的参数进行分页查询,并返回查询结果和分页信息。 ### 回答3: MyBatis是一种持久层框架,可与关系型数据库(如MySQL,Oracle等)进行交互。它通过XML配置或注解的方式,将SQL语句与Java代码进行解耦,简化了数据库操作。 要实现分页查询,我们需要使用MyBatis提供的分页插件(如PageHelper),其具体步骤如下: 1. 在项目的pom.xml文件中添加PageHelper依赖: ```xml <dependency> <groupId>com.github.pagehelper</groupId> <artifactId>pagehelper</artifactId> <version>x.x.x</version> </dependency> ``` 2. 在MyBatis的配置文件中配置PageHelper插件: ```xml <plugins> <plugin interceptor="com.github.pagehelper.PageInterceptor"> <property name="dialect" value="mysql"/> </plugin> </plugins> ``` 3. 在Java代码中使用分页查询功能: ```java // 引入PageHelper类 import com.github.pagehelper.PageHelper; ... // 设置分页参数,pageNum表示当前页码,pageSize表示每页的记录数 PageHelper.startPage(pageNum, pageSize); // 执行查询语句 List<YourEntity> resultList = yourMapper.selectByCondition(condition); // 获取分页信息 PageInfo<YourEntity> pageInfo = new PageInfo<>(resultList); int total = (int) pageInfo.getTotal(); // 总记录数 int pages = pageInfo.getPages(); // 总页数 // 打印分页结果 for (YourEntity entity : resultList) { System.out.println(entity.toString()); } ``` 上述代码中,我们首先调用`PageHelper.startPage(pageNum, pageSize)`方法设置分页参数,然后执行查询语句,得到查询结果。最后,我们可以使用PageInfo对象获取分页的相关信息,例如总记录数和总页数。 需要注意的是,以MySQL为例,在MyBatis中配置PageHelper时需要指定`<property name="dialect" value="mysql"/>`,以确保分页查询语句的正确执行。 总之,通过使用MyBatis提供的分页插件,我们可以很方便地实现分页查询功能,大大简化了开发过程。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值