2

我有一个使用 SODA 包来操作 JSON 数据库的 PL/SQL 处理程序。我想要:

  1. 读取id有效载荷中键的值
  2. 将有效负载 JSON 写入数据库中的新文档。

要执行第 1 步,处理程序需要:body被解析为 JSON_OBJECT_T 类型,以便我可以访问id有效负载中键的值。

但是对于第 2 步,如果我body在构建时再次阅读SODA_DOCUMENT_T(b_content=> :body),它只会给我一个空白文档。

我也不能直接传入 JSON_OBJECT_T 变量SODA_DOCUMENT_T(j_content=> jso),因为该函数需要 JSON 类型而不是 JSON_OBJECT_T。我找不到 JSON 类型的文档,但在代码示例中看到了JSON('{}')生成一个的函数。

然而,阅读:body_text给我带来了其他问题 - 因为JSON()函数无法处理有效负载中的换行符并给出错误。

目前要解决这个问题,我正在使用以下内容:

SODA_DOCUMENT_T(
  j_content => JSON(jso.to_string())
)

这似乎很愚蠢,因为我在将其转换回 JSON 类型之前再次将其序列化为字符串。是否有正确的方法来读取有效负载的键值,并将其传递到 SODA_DOCUMENT_T 中?

4

1 回答 1

3

是的,您不能将JSON_Object_T实例传递给SODA_Document_T构造函数。但是,您可以使用JSON_QUERY()PL/SQL 函数钻入给定路径表达式的 JSON 文档并返回 JSON 类型实例。

例子:

jval := JSON_QUERY(body, '$.id' RETURNING JSON);

上例中,$.id是路径表达式,返回从根开始json_query()的字段对应的json值id$

文档: https ://docs.oracle.com/en/database/oracle/oracle-database/21/adjsn/function-JSON_QUERY.html

您现在应该能够将此jval类型的实例传递JSONSODA_Document_T构造函数。

是的,我认为您绝对应该避免来回转换。

用法:

SQL> 
CREATE OR REPLACE FUNCTION process_request(body  IN VARCHAR2)
RETURN JSON
IS
  j  JSON;
  d  SODA_Document_T;
BEGIN
  
  j := JSON_QUERY(body, '$.id' RETURNING JSON);
  d := SODA_Document_T(j_Content => j);
  -- n := coll.insert_one(d);
  RETURN j;
END;
 13  /

Function created.

SQL> SELECT process_request('{"id":{"type": "string", "val": "DEADBEEF"}}') FROM dual;

PROCESS_REQUEST('{"ID":{"TYPE":"STRING","VAL":"DEADBEEF"}}')
--------------------------------------------------------------------------------
{"type":"string","val":"DEADBEEF"}

我推荐这种方法。让我知道这是否有帮助!

替代解决方案:

顺便说一句,给定一个 DOM,您也可以直接转到 JSON 类型实例,在JSON_Object_T类型中使用以下方法:

MEMBER FUNCTION GET_JSON RETURNS JSON
 Argument Name                  Type                    In/Out Default?
 ------------------------------ ----------------------- ------ --------
 KEY                            VARCHAR2                IN

这种方法与上面所示的工作几乎相同JSON_Query()

SQL>
CREATE OR REPLACE FUNCTION process_body(data IN VARCHAR2)
RETURN JSON
AS
  body JSON_Object_T;
  j    JSON;
BEGIN
  body := JSON_Object_T.parse(data);
  j := body.get_Json('id');
  return j;
END;
 11  /

SQL> SELECT process_body('{"id":{"type": "string", "val": "DEADBEEF"}}') FROM dual;

PROCESS_BODY('{"ID":{"TYPE":"STRING","VAL":"DEADBEEF"}}')
--------------------------------------------------------------------------------
{"type":"string","val":"DEADBEEF"}

于 2022-01-07T00:38:39.403 回答