如何将 Oracle 的 JSON_VALUE 函数与 PreparedStatement 一起使用

Posted

技术标签:

【中文标题】如何将 Oracle 的 JSON_VALUE 函数与 PreparedStatement 一起使用【英文标题】:How to use Oracle's JSON_VALUE function with a PreparedStatement 【发布时间】:2019-07-09 08:05:04 【问题描述】:

我正在尝试使用 Oracle 的 json_value() 函数使用 PreparedStatement 运行 SQL 查询。

假设如下表设置:

drop table foo cascade constraints purge;
create table foo
(
  id integer primary key, 
  payload clob, 
  constraint ensure_json check (payload IS JSON STRICT)
);

insert into foo values (1, '"data": "k1": 1, "k2": "foo"');

以下 SQL 查询工作正常:

select *
from foo
where json_value(payload, '$.data.k1') = '1'

并返回预期的行。

但是,当我尝试使用 PreparedStatement 运行此查询时,如下代码所示:

String sql =
     "select *\n" +
     "from foo\n" +
     "where json_value(payload, ?) = ?";

PreparedStatement pstmt = conection.prepareStatement(sql);
pstmt.setString(1, "$.data.k1");
pstmt.setString(2, "1");
ResultSet rs = pstmt.executeQuery();

(为了简单起见,我从示例中删除了所有错误检查)

这会导致:

java.sql.SQLException: ORA-40454: 路径表达式不是文字

罪魁祸首是传json路径值(参数索引1),第二个参数没问题。

当我(仅)用字符串常量 json_value(payload, '$.data.k1') = ? 替换第一个参数时,准备好的语句可以正常工作。

在绝望的尝试中,我还尝试在参数中包含单引号:pstmt.setString(1, "'$.data.k1'"),但毫不奇怪,Oracle 也不接受它(同样的错误消息)。

我也尝试使用 json_value(payload, concat('$.', ?) ) 并且只传递 "data.k1" 作为参数 - 结果相同。

所以,问题是:

如何使用PreparedStatement 参数将JSON 路径表达式传递给Oracle 的json_value 函数?

有什么想法吗?这是驱动程序中的错误还是 Oracle 中的错误? (我在 My Oracle Support 上找不到任何东西)

或者这仅仅是“未实施”的情况?


环境:

我正在使用 Oracle 18.0 我尝试了 18.3 和 19.3 版本的 ojdbc10.jar 驱动程序以及 OpenJDK 11。

【问题讨论】:

【参考方案1】:

这不是驱动程序 - 你得到同样的东西with dynamic SQL:

declare
  result foo%rowtype;
begin
  execute immediate 'select *
    from foo
    where json_value(payload, :1) = :2'
  into result using '$.data.k1', '1';
  dbms_output.put_line(result.payload);
end;
/

ORA-40454: path expression not a literal
ORA-06512: at line 4

这并不是真正的错误,它是documented(已添加重点):

JSON_basic_path_expression

使用此子句指定 SQL/JSON 路径表达式。该函数使用路径表达式计算 expr 并找到匹配或满足路径表达式的标量 JSON 值。 路径表达式必须是文本文字。请参阅 Oracle Database JSON Developer's Guide 了解 JSON_basic_path_expression 的完整语义

不幸的是,你必须embed the path literal,而不是绑定它:

declare
  result foo%rowtype;
begin
  execute immediate 'select *
    from foo
    where json_value(payload, ''' || '$.data.k1' || ''') = :1'
  into result using '1';
  dbms_output.put_line(result.payload);
end;
/

1 rows affected

dbms_output:
"data": "k1": 1, "k2": "foo"

或者对于您的 JDBC 示例(将路径保留为单独的字符串,因为您可能希望它真的是一个变量):

String sql =
     "select *\n" +
     "from foo\n" +
     "where json_value(payload, '" + "$.data.k1" + "') = ?";

PreparedStatement pstmt = conection.prepareStatement(sql);
pstmt.setString(1, "1");
ResultSet rs = pstmt.executeQuery();

这显然不是您想要做的*,但似乎没有其他选择。除了将您的查询转换为函数并将路径变量传递给该函数之外,该函数必须使用动态 SQL,因此效果大致相同——尽管这样可能更容易处理 SQL 注入问题。

* 而且我知道您知道如何以嵌入式方式执行此操作,并且知道您想使用绑定变量,因为这是正确的做法;我已经把它说得比 需要其他访客 *8-)

【讨论】:

感谢您确认(我没有在手册中发现该部分) - Oracle 的 JSON 支持的另一个(许多)令人讨厌的限制

以上是关于如何将 Oracle 的 JSON_VALUE 函数与 PreparedStatement 一起使用的主要内容,如果未能解决你的问题,请参考以下文章

在 Oracle 12c 的 JSON_VALUE 中使用特殊字符的问题

多个节点同名时使用 Oracle SQL 获取 JSON_VALUE

将具有 JSON 字段的选择结果转换为 JSON,并将该数据与 JSON_VALUE() 一起使用

SQL Server JSON_VALUE 语法

当键的名称中有点时(在 oracle 中),我如何访问 JSON 值?

SQL JSON_VALUE 查询但没有路径名?