Oracle中使用Order By排序时顺序问题

记录一下工作中遇到的问题,虽然说用Oracle的人比较少,但是遇到问题了还是得记录下~

问题

在SQL语句查询中,我们常常要根据某个字段排序来进行查询。如果是MySQL之类的数据库,简单的写一条分页查询语句:

select * from table ORDER BY LIST_ORDER

按照上面的语句,我们就能从table这个表中取出数据并且按照LIST_ORDER字段升序排序(默认升序)。

如果我们再加上分页操作,我们就可以根据根据页数和大小来控制返回的数据量了:

SELECT * FROM (SELECT t.*, ROWNUM AS rowno FROM (select * from table ORDER BY LIST_ORDER) t WHERE ROWNUM<#endRow# ) WHERE ROWNUM>=#startRow#

但是如果在Oracle数据库中进行这样的分页操作,会让返回的数据变得混乱起来。**简单来说就是,你创建了一个分页接口,调用这个接口返回的数据一直有部分数据获取不到,并且获取到的数据有重复内容。**这样的问题在实际工作生产中是非常严重的!!!

那么,怎么解决这个问题呢?

解决办法就是在排序的最后一个字段后面加上主键或者ROWID(ROWNUM)的排序即可解决该问题。

SELECT * FROM (SELECT t.*, ROWNUM AS rowno FROM (select * from table ORDER BY LIST_ORDER, ROWNUM) t WHERE ROWNUM<#endRow# ) WHERE ROWNUM>=#startRow#

原因

为什么会导致这个问题?

我看到的一个说法是,Oracle在执行分页操作的时候,其实分了四步:

  1. 根据SQL语句查询数据
  2. 给查询到的数据每一行都添加ROWNUM
  3. Order By排序
  4. 分页处理

这里就存在一个问题,与MySQL数据库不同,分页操作多了第二步,就意味着下一步进行Order By排序的时候,处理后的数据的ROWNUM会丢失顺序,而ORACLE的分页处理是根据这个ROWNUM进行操作的,一旦ROWNUM混乱,那么返回的数据也会混乱(包括返回的数据丢失,或者不同页之间有重复内容)。

解决的方法其实就是像上面说的一样,Order By后面加个ROWNUM即可解决问题。

你可能感兴趣的:(oracle)