0

0

如何使用Java对Oracle进行分页查询_ROWNUM与FETCH NEXT特性的JDBC兼容性

P粉602998670

P粉602998670

发布时间:2026-03-17 08:48:13

|

538人浏览过

|

来源于php中文网

原创

Oracle 12c+ 分页需用 FETCH NEXT,但要求 JDBC 驱动 ≥12.1.0.2(如 ojdbc8.jar),低版本会报 ORA-00933;ROWNUM 方式必须两层嵌套且 ORDER BY 在最内层;FETCH NEXT 参数须用 setLong 绑定,排序含 NULL 时需显式声明 NULLS LAST。

Oracle 12c+ 用 FETCH NEXT,但 JDBC 驱动版本太低会报错

java 里直接写 select * from t order by id fetch next 10 rows only 看似干净,实际运行时可能抛出 ora-00933: sql command not properly ended——这不是 sql 写错了,是 jdbc 驱动不认这个语法。oracle 12c 引入的 fetch next 需要 oracle jdbc driver 12.1.0.2 或更高版本才支持。低于该版本(比如常见的 ojdbc6.jar)会把 fetch 当作非法关键字。

实操建议:

  • 检查当前项目使用的 JAR 包:看文件名是否含 ojdbc8.jarojdbc11.jar;若为 ojdbc6.jarojdbc7.jar,必须升级
  • 升级后仍报错?确认 Connection 没有被旧驱动缓存——重启应用、清空 classloader、删掉 IDE 的 output/cache 目录
  • ojdbc8.jar 支持 JDK 8+,且默认开启对 FETCH NEXT 的解析,无需额外配置

ROWNUM 嵌套写法必须带排序子查询,否则分页结果错乱

很多人照搬网上模板写 SELECT * FROM (SELECT ROWNUM rn, t.* FROM table t) WHERE rn BETWEEN ? AND ?,结果发现第 2 页数据和第 1 页重复,或顺序随机。根本原因是 ROWNUM 在 Oracle 中是“取行时分配”的,它不等价于“按某列排序后的序号”。没加 ORDER BY 的子查询,Oracle 可能每次返回不同物理顺序,ROWNUM 就跟着飘。

正确写法必须两层嵌套,且 ORDER BY 要落在最内层:

SELECT * FROM (
  SELECT ROWNUM rn, t.* FROM (
    SELECT * FROM users ORDER BY created_time DESC
  ) t
) WHERE rn BETWEEN 11 AND 20

常见错误:

立即学习Java免费学习笔记(深入)”;

  • ORDER BY 放在外层:无效,ROWNUM 已经分配完了
  • 子查询里漏了括号或别名,导致语法错误或逻辑错位
  • ROWNUM <= N 单边过滤(如 WHERE ROWNUM <= 20),无法跳过前 N 行,只能用于取前 N 条

JDBC PreparedStatement 绑定 FETCH NEXT 参数要小心位置和类型

FETCH NEXT ? ROWS ONLY 看起来能用 PreparedStatement.setLong(1, 10),但 Oracle JDBC 对这个位置参数的支持在早期 ojdbc8 版本中并不稳定——某些补丁版本会把 ? 当作文本字面量处理,导致 ORA-01785(ORDER BY 项必须在 SELECT 列表中)这类误报。

更稳妥的做法是:参数只用于偏移量和数量,不用于 FETCH 关键字本身。例如:

SELECT * FROM orders ORDER BY id 
OFFSET ? ROWS FETCH NEXT ? ROWS ONLY

这时两个 ? 都必须用 setLong(不能用 setInt),因为 Oracle 内部将偏移量和行数视为 NUMBER 类型,JDBC 驱动在类型推断失败时可能截断或报 SQLException

NameGPT名称生成器
NameGPT名称生成器

免费AI公司名称生成器,AI在线生成企业名称,注册公司名称起名大全。

下载

注意点:

  • 第一个参数(OFFSET)可以为 0,但不能为负数,否则报 ORA-01785
  • 如果业务允许,优先用 setLong(1, (page - 1L) * size) 计算偏移,避免整数溢出
  • 不要在同一个 SQL 里混用 ROWNUMFETCH,驱动解析行为未定义

分页字段含 NULL 时,ORDER BY 必须显式声明 NULLS LAST

Oracle 默认把 NULL 排在最前面(NULLS FIRST),而多数业务期望非空值优先展示。如果你分页查一个状态字段(如 status),其中大量为 NULL,又没加 NULLS LAST,第 1 页可能全是 NULL 行,用户以为数据为空。

这个问题在 ROWNUMFETCH NEXT 两种方式下都存在,但 FETCH NEXT 更隐蔽——因为语法看起来“现代”,容易忽略排序细节。

正确写法示例:

SELECT * FROM products 
ORDER BY status NULLS LAST, id DESC 
OFFSET 0 ROWS FETCH NEXT 10 ROWS ONLY

关键提醒:

  • NULLS FIRST/LAST 是 Oracle 特有语法,MySQL/PostgreSQL 不兼容,换库时要重写
  • 如果排序字段建了索引,确保索引也包含 NULLS LAST 语义(Oracle 12c+ 支持函数索引如 CREATE INDEX idx ON t(status, id) NULLS LAST
  • ROWNUM 方式时,NULLS LAST 必须写在最内层子查询的 ORDER BY 中,否则无效

Oracle 分页真正难的不是语法,是驱动版本、NULL 排序、参数类型这三处细节叠在一起时,错误现象和真实原因完全不匹配。调半天发现是 jar 包太老,或者 ORDER BY 少写了 NULLS LAST,这种事很常见。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

1135

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

340

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

381

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

2257

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

380

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

1763

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

587

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

441

2024.04.29

chatgpt使用指南
chatgpt使用指南

本专题整合了chatgpt使用教程、新手使用说明等等相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.03.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号