Oracle分页
2015-11-09 22:57
288 查看
先看以下两条语句的执行结果:
语句一:select rownum,empno,sal from emp order by empno;
ROWNUM EMPNO SAL
---------- ---------- ----------
1 7369 800
2 7499 1600
3 7521 1250
4 7566 2975
5 7654 1250
6 7698 2850
7 7782 2450
8 7788 3000
语句二:select rownum,empno,sal from emp order by sal;
ROWNUM EMPNO SAL
---------- ---------- ----------
1 7369 800
12 7900 950
11 7876 1100
3 7521 1250
5 7654 1250
14 7934 1300
10 7844 1500
2 7499 1600
同样的两个语句,执行结果中的rownum伪列的值却大相径庭,这是什么原因呢?其实这都是ORACLE内部的查询优化器和索引搞的鬼!
rownum伪列产生的序号是按照数据被查询出来的顺序添加上去的,第一条是1,第二条是2,依次加1。
当将一条语句交给查询优化器处理时:
如果排序列上有索引,则借助索引去查询数据,这样,读取出来的数据和rownum产生的序号是一种正常的对应关系,如语句一的结果(empno上有主键索引)。
如果排序列上没有索引,则使用全表扫描的方式,依次从表中读取数据,读取完成后,最后进行排序,于是产生了类似语句二的结果(sal列上没有索引)。
正是由于排序列上不一定有索引,所以在ORACLE中使用rownum伪列分页时,需要多加一层查询,以保证rownum序号的连续性。
语句三:select rownum,t.* from (select empno,sal from emp order by sal) t;
ROWNUM EMPNO SAL
---------- ---------- ----------
1 7369 800
2 7900 950
3 7876 1100
4 7521 1250
5 7654 1250
6 7934 1300
7 7844 1500
这个结果还满意吧。。。
分页:在外面再加上一层查询。
select * from (select rownum num,t.* from (select empno,sal from emp order by sal) t) where num between 6 and 10;
当然,如果使用分析函数row_number就可以省略一层查询了,代码更简单点:
select * from (select row_number() over (order by sal) num,empno,sal from emp) where num between 6 and 10;
语句一:select rownum,empno,sal from emp order by empno;
ROWNUM EMPNO SAL
---------- ---------- ----------
1 7369 800
2 7499 1600
3 7521 1250
4 7566 2975
5 7654 1250
6 7698 2850
7 7782 2450
8 7788 3000
语句二:select rownum,empno,sal from emp order by sal;
ROWNUM EMPNO SAL
---------- ---------- ----------
1 7369 800
12 7900 950
11 7876 1100
3 7521 1250
5 7654 1250
14 7934 1300
10 7844 1500
2 7499 1600
同样的两个语句,执行结果中的rownum伪列的值却大相径庭,这是什么原因呢?其实这都是ORACLE内部的查询优化器和索引搞的鬼!
rownum伪列产生的序号是按照数据被查询出来的顺序添加上去的,第一条是1,第二条是2,依次加1。
当将一条语句交给查询优化器处理时:
如果排序列上有索引,则借助索引去查询数据,这样,读取出来的数据和rownum产生的序号是一种正常的对应关系,如语句一的结果(empno上有主键索引)。
如果排序列上没有索引,则使用全表扫描的方式,依次从表中读取数据,读取完成后,最后进行排序,于是产生了类似语句二的结果(sal列上没有索引)。
正是由于排序列上不一定有索引,所以在ORACLE中使用rownum伪列分页时,需要多加一层查询,以保证rownum序号的连续性。
语句三:select rownum,t.* from (select empno,sal from emp order by sal) t;
ROWNUM EMPNO SAL
---------- ---------- ----------
1 7369 800
2 7900 950
3 7876 1100
4 7521 1250
5 7654 1250
6 7934 1300
7 7844 1500
这个结果还满意吧。。。
分页:在外面再加上一层查询。
select * from (select rownum num,t.* from (select empno,sal from emp order by sal) t) where num between 6 and 10;
当然,如果使用分析函数row_number就可以省略一层查询了,代码更简单点:
select * from (select row_number() over (order by sal) num,empno,sal from emp) where num between 6 and 10;
相关文章推荐
- 存储过程 函数(基础一)
- 47.Oracle数据库SQL开发之 子查询——子查询的类型
- 46.Oracle数据库SQL开发之 日期和时间的存储与处理——使用时间间隔
- 45.Oracle数据库SQL开发之 日期和时间的存储与处理——使用时间戳
- 44.Oracle数据库SQL开发之 日期和时间的存储与处理——理解时区
- 43.Oracle数据库SQL开发之 日期和时间的存储与处理——使用时间值函数
- oracle存储过程中使用其他用户的表或视图
- 如何查看 oracle 官方文档
- oracle参数open_cursors和session_cached_cursor详解!
- Oracle数据库中插入日期型数据
- oracle和其他数据库对表名、列名的长度限制
- oracle listagg函数、lag函数、lead函数 实例
- 关于oracle函数listagg的使用说明
- Oracle CASE WHEN 用法介绍
- oracle恢复删除的数据
- Oracle 阻塞会话的查看与解除
- Oracle 中 sign和decode 函数用法
- 将oracle的Date转换为相应格式输出
- oracle数据库的导出与导入备忘
- Ubuntu Linux下安装Oracle JDK