Oracle查询结果分页没有TABLE FULL SCAN数据访问方法

时间:2015-03-24 12:37:31

标签: oracle hibernate pagination

关于如何正确执行分页的stackoverflow有很多问题,对于oracle,最流行的答案是这样的:

select * from (
    select row_.*, rownum rownum_ 
    from (select * from some_table) row_ 
    where rownum <= N) 
where rownum_ > M;

我到处都看过这个问题,而Hibernate也会为分页生成它。它可能适用于大多数情况,但它需要全表扫描并且会显着减慢大量数据。

有一条提示可以帮助选择前N行

/*+ FIRST_ROWS(5000) */

但是在选择第二页并且似乎也使用全扫描时它没有帮助,至少那是什么&#34;解释计划&#34;对我说。

为了解决这个问题,我目前正在实现自定义分页 - 读取表中所有行的ID并将它们拆分到范围上,以便分页查询看起来像这样:

select * from some_table where id between N and M;

我希望找到供应商解决这个问题的方法,但到目前为止还没有成功。

所以,问题是 - 我是否重新发明轮子,或者在没有完全扫描的情况下真的没有办法在oracle上实现分页?

Upd:在Oracle 12c中,他们引入了一种新的分页语法:

OFFSET N ROWS FETCH NEXT M ROWS ONLY;

我已尝试解释此计划,它似乎只是

的别名
select * from (
    select row_.*, rownum rownum_ 
    from (select * from some_table) row_ 
    where rownum <= N) 
where rownum_ > M;

UPD2:刚发现一个类似的问题 - Oracle and Pagination 看起来我以前在搜索重复项时一直不专心。所以,很可能我的问题的答案是否定的,但是,从那以后可能会发生一些变化......

1 个答案:

答案 0 :(得分:1)

首先:全表扫描并不总是恶魔。

  1. 在您的解释计划中检查没有分页的查询费用
  2. 在您的分页解决方案中查看
  3. 此外,当您进行测试时,请尝试使用具有较高分页值的大型表

    补充要点:

    • 没有订购的分页总是危险的,因为您无法确定,Oracle会以何种顺序为您的下一页&#34;页面提供结果。 - &GT;可能无法重复的结果
    • 同样在有序的结果中,有可能是一个&#34;新条目&#34;在当前&#34;查看&#34;之前的页面中影响您的&#34;下一页&#34;

    我除了,你喜欢&#34;存储&#34; DB中的查询并逐页获取数据,直到下划线数据中的某些内容发生变化?