2025年9月25日: PostgreSQL 18 发布!
支持的版本: 当前 (18) / 17 / 16 / 15 / 14 / 13
开发版本: devel
不支持的版本: 12 / 11 / 10 / 9.6 / 9.5 / 9.4 / 9.3 / 9.2

8.14. JSON类型 #

JSON 数据类型用于存储 JSON (JavaScript Object Notation) 数据,如 RFC 7159 中所指定。此类数据也可以存储为 text,但 JSON 数据类型具有强制要求每个存储的值都符合 JSON 规则的优点。这些数据类型还提供了各种 JSON 特定函数和运算符;请参阅 第 9.16 节

PostgreSQL 提供两种存储 JSON 数据的数据类型:jsonjsonb。为了实现这些数据类型的有效查询机制,PostgreSQL 还提供了 第 8.14.7 节 中描述的 jsonpath 数据类型。

jsonjsonb 数据类型在输入时接受几乎相同的数值集。主要的实际区别在于效率。json 数据类型存储输入的文本的精确副本,处理函数必须在每次执行时重新解析;而 jsonb 数据以分解的二进制格式存储,由于增加了转换开销,输入时稍慢,但处理速度显著提高,因为不需要重新解析。jsonb 还支持索引,这可能是一个重要的优势。

由于 json 类型存储输入的文本的精确副本,因此它会保留令牌之间的语义上不重要的空格,以及 JSON 对象中键的顺序。此外,如果值中的 JSON 对象包含多个相同的键,则会保留所有键/值对。(处理函数将最后一个值视为有效的。)相比之下,jsonb 不会保留空格,也不会保留对象键的顺序,也不会保留重复的对象键。如果输入中指定了重复的键,则只保留最后一个值。

通常,大多数应用程序应倾向于将 JSON 数据存储为 jsonb,除非有非常特殊的需求,例如对对象键排序的遗留假设。

RFC7159 指定 JSON 字符串应使用 UTF8 编码。因此,除非数据库编码为 UTF8,否则 JSON 类型无法严格遵守 JSON 规范。尝试直接包含数据库编码无法表示的字符将失败;相反,允许表示为数据库编码但无法在 UTF8 中表示的字符。

RFC7159 允许 JSON 字符串包含由 \uXXXX 表示的 Unicode 转义序列。在 json 类型的输入函数中,无论数据库编码如何,都允许使用 Unicode 转义,并且仅检查语法正确性(即,\u 后面是四个十六进制数字)。但是,jsonb 的输入函数更严格:它不允许使用数据库编码无法表示的字符的 Unicode 转义。jsonb 类型还会拒绝 \u0000(因为这在 PostgreSQLtext 类型中无法表示),并且它坚持要求使用 Unicode 代理对来表示 Unicode 基本多语言平面之外的字符时必须正确。有效的 Unicode 转义将被转换为等效的单个字符进行存储;这包括将代理对折叠成单个字符。

注意

第 9.16 节 中描述的许多 JSON 处理函数会将 Unicode 转义转换为普通字符,因此即使它们的输入类型是 json 而不是 jsonb,也会抛出与上述相同的错误。事实上,json 输入函数不执行这些检查可能被视为历史遗留,尽管它允许在不支持所表示字符的数据库编码中简单地存储(不处理)JSON Unicode 转义。

将文本 JSON 输入转换为 jsonb 时,7159 所描述的原始类型有效地映射到本地 PostgreSQL 类型,如 表 8.23 所示。因此,对于什么是有效的 jsonb 数据,存在一些与 json 类型或抽象的 JSON 不适用的轻微附加限制,这对应于底层数据类型所能表示的内容的限制。尤其值得注意的是,jsonb 会拒绝超出 PostgreSQL numeric 数据类型范围的数字,而 json 不会。7159 允许此类实现定义的限制。然而,实际上,这类问题在其他实现中出现的可能性要大得多,因为通常将 JSON 的 number 原始类型表示为 IEEE 754 双精度浮点数(7159 明确预期并允许)。在与其他系统使用 JSON 作为交换格式时,应考虑与 PostgreSQL 最初存储的数据相比,丢失数值精度的风险。RFC7159 允许 JSON 字符串包含 Unicode 转义序列,表示为 \uXXXX。在 json 类型的输入函数中,Unicode 转义允许,而不考虑数据库编码,并且仅在语法上进行检查(即,\u 后面是四个十六进制数字)。然而,jsonb 的输入函数更严格:它不允许使用数据库编码无法表示的字符的 Unicode 转义。jsonb 类型还会拒绝 \u0000(因为这在 PostgreSQLtext 类型中无法表示),并且它坚持要求使用 Unicode 代理对来表示 Unicode 基本多语言平面之外的字符时必须正确。有效的 Unicode 转义将被转换为等效的单个字符进行存储;这包括将代理对折叠成单个字符。RFC7159 允许 JSON 字符串包含 Unicode 转义序列,表示为 \uXXXX。在 json 类型的输入函数中,Unicode 转义允许,而不考虑数据库编码,并且仅在语法上进行检查(即,\u 后面是四个十六进制数字)。然而,jsonb 的输入函数更严格:它不允许使用数据库编码无法表示的字符的 Unicode 转义。jsonb 类型还会拒绝 \u0000(因为这在 PostgreSQLtext 类型中无法表示),并且它坚持要求使用 Unicode 代理对来表示 Unicode 基本多语言平面之外的字符时必须正确。有效的 Unicode 转义将被转换为等效的单个字符进行存储;这包括将代理对折叠成单个字符。RFC7159 明确预期并允许 JSON 的 number 原始类型表示为 IEEE 754 双精度浮点数。当使用 JSON 作为与其他系统交换格式时,应考虑与 PostgreSQL 最初存储的数据相比,丢失数值精度的风险。

相反,如表中所示,JSON 原始类型的输入格式存在一些轻微限制,而这些限制不适用于相应的 PostgreSQL 类型。

表 8.23. JSON 原始类型和相应的 PostgreSQL 类型

JSON 原始类型 PostgreSQL 类型 注释
string text \u0000 不允许,包含表示数据库编码中不可用字符的 Unicode 转义也不允许
number numeric NaNinfinity 值不允许
boolean boolean 仅接受小写的 truefalse 拼写
null (无) SQL NULL 是一个不同的概念

8.14.1. JSON 输入和输出语法 #

JSON 数据类型的输入/输出语法如RFC 7159.

以下均为有效的 json(或 jsonb)表达式

-- Simple scalar/primitive value
-- Primitive values can be numbers, quoted strings, true, false, or null
SELECT '5'::json;

-- Array of zero or more elements (elements need not be of same type)
SELECT '[1, 2, "foo", null]'::json;

-- Object containing pairs of keys and values
-- Note that object keys must always be quoted strings
SELECT '{"bar": "baz", "balance": 7.77, "active": false}'::json;

-- Arrays and objects can be nested arbitrarily
SELECT '{"foo": [true, "bar"], "tags": {"a": 1, "b": null}}'::json;

如前所述,当 JSON 值被输入然后打印而没有任何额外处理时,json 输出的文本与输入的相同,而 jsonb 不会保留语义上不重要的细节,例如空格。例如,请注意这里的区别

SELECT '{"bar": "baz", "balance": 7.77, "active":false}'::json;
                      json
-------------------------------------------------
 {"bar": "baz", "balance": 7.77, "active":false}
(1 row)

SELECT '{"bar": "baz", "balance": 7.77, "active":false}'::jsonb;
                      jsonb
--------------------------------------------------
 {"bar": "baz", "active": false, "balance": 7.77}
(1 row)

一个值得注意的语义上不重要的细节是,在 jsonb 中,数字将根据底层 numeric 类型的行为进行打印。实际上,这意味着带有 E 表示法的数字将不带 E 表示法打印,例如

SELECT '{"reading": 1.230e-5}'::json, '{"reading": 1.230e-5}'::jsonb;
         json          |          jsonb
-----------------------+-------------------------
 {"reading": 1.230e-5} | {"reading": 0.00001230}
(1 row)

然而,jsonb 会保留尾随的零分数,如本例所示,即使这些对于相等性检查等目的而言在语义上不重要。

有关用于构造和处理 JSON 值的内置函数和运算符列表,请参阅 第 9.16 节

8.14.2. 设计 JSON 文档 #

将数据表示为 JSON 可能比传统的二维数据模型灵活得多,这在需求不断变化的坏境中很有吸引力。这两种方法在同一个应用程序中共存和互补是完全可能的。但是,即使对于需要最大灵活性的应用程序,仍然建议 JSON 文档具有一定的固定结构。结构通常不受强制(尽管可以声明性地强制执行某些业务规则),但具有可预测的结构使得编写有用地汇总表中一组文档(数据项)的查询变得更加容易。

当存储在表中时,JSON 数据与其他任何数据类型一样会受到相同的并发控制考虑。虽然存储大型文档是可行的,但请记住,任何更新都会获取对整个行的行级锁。考虑将 JSON 文档限制在可管理的大小,以减少更新事务之间的锁定争用。理想情况下,JSON 文档应代表业务规则规定无法合理地进一步细分为可独立修改的更小数据项的原子数据项。

8.14.3. jsonb 包含和存在性 #

包含测试是 jsonb 的一项重要功能。对于 json 类型没有类似的设施。包含测试一个 jsonb 文档是否包含另一个文档。除了注明的情况外,这些示例都返回 true。

-- Simple scalar/primitive values contain only the identical value:
SELECT '"foo"'::jsonb @> '"foo"'::jsonb;

-- The array on the right side is contained within the one on the left:
SELECT '[1, 2, 3]'::jsonb @> '[1, 3]'::jsonb;

-- Order of array elements is not significant, so this is also true:
SELECT '[1, 2, 3]'::jsonb @> '[3, 1]'::jsonb;

-- Duplicate array elements don't matter either:
SELECT '[1, 2, 3]'::jsonb @> '[1, 2, 2]'::jsonb;

-- The object with a single pair on the right side is contained
-- within the object on the left side:
SELECT '{"product": "PostgreSQL", "version": 9.4, "jsonb": true}'::jsonb @> '{"version": 9.4}'::jsonb;

-- The array on the right side is not considered contained within the
-- array on the left, even though a similar array is nested within it:
SELECT '[1, 2, [1, 3]]'::jsonb @> '[1, 3]'::jsonb;  -- yields false

-- But with a layer of nesting, it is contained:
SELECT '[1, 2, [1, 3]]'::jsonb @> '[[1, 3]]'::jsonb;

-- Similarly, containment is not reported here:
SELECT '{"foo": {"bar": "baz"}}'::jsonb @> '{"bar": "baz"}'::jsonb;  -- yields false

-- A top-level key and an empty object is contained:
SELECT '{"foo": {"bar": "baz"}}'::jsonb @> '{"foo": {}}'::jsonb;

一般原则是,被包含对象必须在结构和数据内容上匹配包含对象,可能在丢弃包含对象中一些不匹配的数组元素或对象键/值对之后。但请记住,在进行包含匹配时,数组元素的顺序不重要,重复的数组元素实际上只考虑一次。

作为一般原则的特殊例外,结构必须匹配,一个数组可以包含一个原始值

-- This array contains the primitive string value:
SELECT '["foo", "bar"]'::jsonb @> '"bar"'::jsonb;

-- This exception is not reciprocal -- non-containment is reported here:
SELECT '"bar"'::jsonb @> '["bar"]'::jsonb;  -- yields false

jsonb 还有一个存在性运算符,它是包含主题的一个变体:它测试一个字符串(以 text 值给出)是否出现在 jsonb 值顶层的对象键或数组元素中。除了注明的情况外,这些示例都返回 true。

-- String exists as array element:
SELECT '["foo", "bar", "baz"]'::jsonb ? 'bar';

-- String exists as object key:
SELECT '{"foo": "bar"}'::jsonb ? 'foo';

-- Object values are not considered:
SELECT '{"foo": "bar"}'::jsonb ? 'bar';  -- yields false

-- As with containment, existence must match at the top level:
SELECT '{"foo": {"bar": "baz"}}'::jsonb ? 'bar'; -- yields false

-- A string is considered to exist if it matches a primitive JSON string:
SELECT '"foo"'::jsonb ? 'foo';

当涉及许多键或元素时,JSON 对象比数组更适合测试包含或存在性,因为与数组不同,它们在内部经过优化以便搜索,并且不需要线性搜索。

提示

由于 JSON 包含是嵌套的,因此合适的查询可以跳过显式选择子对象。例如,假设我们有一个 doc 列,其中包含顶层的对象,大多数对象包含包含子对象数组的 tags 字段。此查询查找包含 "term":"paris""term":"food" 的子对象的条目,同时忽略 tags 数组之外的任何此类键。

SELECT doc->'site_name' FROM websites
  WHERE doc @> '{"tags":[{"term":"paris"}, {"term":"food"}]}';

可以使用类似的方法完成相同的事情,例如

SELECT doc->'site_name' FROM websites
  WHERE doc->'tags' @> '[{"term":"paris"}, {"term":"food"}]';

但这种方法不太灵活,而且通常效率也较低。

另一方面,JSON 存在性运算符不是嵌套的:它只查找 JSON 值顶层的指定键或数组元素。

各种包含和存在性运算符以及所有其他 JSON 运算符和函数都记录在 第 9.16 节 中。

8.14.4. jsonb 索引 #

GIN 索引可用于高效地搜索大量 jsonb 文档(数据项)中出现的键或键/值对。提供了两个 GIN运算符类,提供不同的性能和灵活性权衡。

用于 jsonb 的默认 GIN 运算符类支持使用键存在运算符 ??|?&、包含运算符 @> 以及 jsonpath 匹配运算符 @?@@ 的查询。(有关这些运算符实现的语义的详细信息,请参阅 表 9.48。)创建此运算符类的索引的示例是

CREATE INDEX idxgin ON api USING GIN (jdoc);

非默认 GIN 运算符类 jsonb_path_ops 不支持键存在运算符,但它支持 @>@?@@。创建此运算符类的索引的示例是

CREATE INDEX idxginp ON api USING GIN (jdoc jsonb_path_ops);

考虑一个存储从第三方 Web 服务检索到的 JSON 文档的表,该表具有文档化的架构定义。典型的文档是

{
    "guid": "9c36adc1-7fb5-4d5b-83b4-90356a46061a",
    "name": "Angela Barton",
    "is_active": true,
    "company": "Magnafone",
    "address": "178 Howard Place, Gulf, Washington, 702",
    "registered": "2009-11-07T08:53:22 +08:00",
    "latitude": 19.793713,
    "longitude": 86.513373,
    "tags": [
        "enim",
        "aliquip",
        "qui"
    ]
}

我们将这些文档存储在名为 api 的表中,在名为 jdocjsonb 列中。如果在该列上创建了 GIN 索引,则以下查询可以使用该索引。

-- Find documents in which the key "company" has value "Magnafone"
SELECT jdoc->'guid', jdoc->'name' FROM api WHERE jdoc @> '{"company": "Magnafone"}';

但是,索引不能用于以下类型的查询,因为尽管运算符 ? 可索引,但它没有直接应用于索引列 jdoc

-- Find documents in which the key "tags" contains key or array element "qui"
SELECT jdoc->'guid', jdoc->'name' FROM api WHERE jdoc -> 'tags' ? 'qui';

尽管如此,通过适当使用表达式索引,上述查询可以使用索引。如果经常查询 "tags" 键内的特定项,则定义如下索引可能值得。

CREATE INDEX idxgintags ON api USING GIN ((jdoc -> 'tags'));

现在,WHERE 子句 jdoc -> 'tags' ? 'qui' 将被识别为在索引表达式 jdoc -> 'tags' 上应用了可索引运算符 ?。(有关表达式索引的更多信息,请参阅 第 11.7 节。)

另一种查询方法是利用包含,例如

-- Find documents in which the key "tags" contains array element "qui"
SELECT jdoc->'guid', jdoc->'name' FROM api WHERE jdoc @> '{"tags": ["qui"]}';

jdoc 列上的简单 GIN 索引可以支持此查询。但请注意,此类索引将存储 jdoc 列中的每个键和值的副本,而前一个示例中的表达式索引仅存储 tags 键下的数据。虽然简单的索引方法更灵活(因为它支持关于任何键的查询),但有针对性的表达式索引可能比简单索引更小,搜索速度更快。

GIN 索引还支持 @?@@ 运算符,它们执行 jsonpath 匹配。示例是

SELECT jdoc->'guid', jdoc->'name' FROM api WHERE jdoc @? '$.tags[*] ? (@ == "qui")';
SELECT jdoc->'guid', jdoc->'name' FROM api WHERE jdoc @@ '$.tags[*] == "qui"';

对于这些运算符,GIN 索引会从 jsonpath 模式中提取 accessors_chain == constant 形式的子句,并基于这些子句中提到的键和值执行索引搜索。访问器链可以包括 .key[*][index] 访问器。jsonb_ops 运算符类还支持 .*.** 访问器,但 jsonb_path_ops 运算符类不支持。

尽管 jsonb_path_ops 运算符类仅支持使用 @>@?@@ 运算符的查询,但它与默认运算符类 jsonb_ops 相比具有显著的性能优势。jsonb_path_ops 索引通常比相同数据的 jsonb_ops 索引小得多,并且搜索的特异性更好,尤其是在查询包含数据中经常出现的键时。因此,搜索操作通常比默认运算符类更好。

jsonb_opsjsonb_path_ops GIN 索引之间,技术上的区别在于前者为数据中的每个键和值创建独立的索引项,而后者仅为数据中的每个值创建索引项。[7] 基本上,每个 jsonb_path_ops 索引项是值和导致它的键(们)的哈希值;例如,要索引 {"foo": {"bar": "baz"}},将创建一个包含 foobarbaz 的单个索引项到哈希值中。因此,查找此结构的包含查询将导致非常特定的索引搜索;但没有办法确定 foo 是否作为键出现。另一方面,jsonb_ops 索引将创建三个单独表示 foobarbaz 的索引项;然后,要执行包含查询,它将查找包含这三个项的行。虽然 GIN 索引可以相当有效地执行这种 AND 搜索,但它仍然不如等效的 jsonb_path_ops 搜索,特别是如果存在包含这三个索引项中任何一个的非常多的行。

jsonb_path_ops 方法的一个缺点是它不会为不包含任何值的 JSON 结构创建索引条目,例如 {"a": {}}。如果请求搜索包含此类结构的文档,将需要一个完整的索引扫描,这非常慢。jsonb_path_ops 因此不适合经常执行此类搜索的应用程序。

jsonb 还支持 btreehash 索引。这些通常仅在检查完整 JSON 文档的相等性很重要时才有用。jsonb 数据项的 btree 排序很少有很大的意义,但为完整起见,它是

Object > Array > Boolean > Number > String > null

Object with n pairs > object with n - 1 pairs

Array with n elements > array with n - 1 elements

例外情况是(由于历史原因)顶层空数组排序小于null。具有相同数量对的对象按顺序比较

key-1, value-1, key-2 ...

请注意,对象键按其存储顺序进行比较;特别是,因为较短的键比较长的键先存储,这可能导致一些可能令人费解的结果,例如

{ "aa": 1, "c": 1} > {"b": 1, "d": 1}

同样,具有相同数量元素的数组按顺序比较

element-1, element-2 ...

原始 JSON 值使用与底层 PostgreSQL 数据类型相同的比较规则进行比较。字符串使用默认数据库排序规则进行比较。

8.14.5. jsonb 下标 #

jsonb 数据类型支持类数组的下标表达式来提取和修改元素。可以通过链接下标表达式来指示嵌套值,遵循与 jsonb_set 函数中的 path 参数相同的规则。如果 jsonb 值是数组,则数字下标从零开始,负整数从数组最后一个元素向后计数。不支持切片表达式。下标表达式的结果始终是 jsonb 数据类型。

UPDATE 语句可以在 SET 子句中使用下标来修改 jsonb 值。下标路径必须对于所有受影响的值(只要它们存在)都可以遍历。例如,路径 val['a']['b']['c'] 可以一直遍历到 c,如果每个 valval['a']val['a']['b'] 都是对象。如果任何 val['a']val['a']['b'] 未定义,它将被创建为空对象并按需填充。但是,如果任何 val 本身或中间值被定义为非对象(如字符串、数字或 jsonb null),则遍历无法继续,将引发错误并中止事务。

下标语法的示例

-- Extract object value by key
SELECT ('{"a": 1}'::jsonb)['a'];

-- Extract nested object value by key path
SELECT ('{"a": {"b": {"c": 1}}}'::jsonb)['a']['b']['c'];

-- Extract array element by index
SELECT ('[1, "2", null]'::jsonb)[1];

-- Update object value by key. Note the quotes around '1': the assigned
-- value must be of the jsonb type as well
UPDATE table_name SET jsonb_field['key'] = '1';

-- This will raise an error if any record's jsonb_field['a']['b'] is something
-- other than an object. For example, the value {"a": 1} has a numeric value
-- of the key 'a'.
UPDATE table_name SET jsonb_field['a']['b']['c'] = '1';

-- Filter records using a WHERE clause with subscripting. Since the result of
-- subscripting is jsonb, the value we compare it against must also be jsonb.
-- The double quotes make "value" also a valid jsonb string.
SELECT * FROM table_name WHERE jsonb_field['key'] = '"value"';

jsonb 通过下标进行的赋值在某些边缘情况下与 jsonb_set 不同。当源 jsonb 值为 NULL 时,通过下标赋值将按其类型(对象或数组)的空 JSON 值进行,该类型由下标键暗示。

-- Where jsonb_field was NULL, it is now {"a": 1}
UPDATE table_name SET jsonb_field['a'] = '1';

-- Where jsonb_field was NULL, it is now [1]
UPDATE table_name SET jsonb_field[0] = '1';

如果为元素数量不足的数组指定了索引,将附加 NULL 元素,直到可以访问索引并设置值。

-- Where jsonb_field was [], it is now [null, null, 2];
-- where jsonb_field was [0], it is now [0, null, 2]
UPDATE table_name SET jsonb_field[2] = '2';

jsonb 值将接受对不存在的下标路径的赋值,只要要遍历的最后一个现有元素是对象或数组(由相应的下标暗示,最后一个下标路径中的元素不遍历,可以是任何值)。将创建嵌套的数组和对象结构,并且在前者情况下,将根据下标路径填充 null,直到可以放置赋值的值。

-- Where jsonb_field was {}, it is now {"a": [{"b": 1}]}
UPDATE table_name SET jsonb_field['a'][0]['b'] = '1';

-- Where jsonb_field was [], it is now [null, {"a": 1}]
UPDATE table_name SET jsonb_field[1]['a'] = '1';

8.14.6. 转换 #

还有额外的扩展实现了不同过程语言的 jsonb 类型的转换。

PL/Perl 的扩展称为 jsonb_plperljsonb_plperlu。如果使用它们,jsonb 值将根据情况映射到 Perl 数组、哈希和标量。

PL/Python 的扩展称为 jsonb_plpython3u。如果使用它,jsonb 值将根据情况映射到 Python 字典、列表和标量。

在这些扩展中,jsonb_plperl 被认为是受信任的,即,它可以由没有数据库超级用户权限但具有当前数据库CREATE权限的非超级用户安装。其余扩展需要超级用户权限才能安装。

8.14.7. jsonpath 类型 #

jsonpath 类型在 PostgreSQL 中实现了对 SQL/JSON 路径语言的支持,以便高效地查询 JSON 数据。它提供了已解析的 SQL/JSON 路径表达式的二进制表示,该表达式指定路径引擎从 JSON 数据中检索要进一步与 SQL/JSON 查询函数进行处理的项目。

SQL/JSON 路径谓词和运算符的语义通常遵循 SQL。同时,为了提供一种处理 JSON 数据的自然方式,SQL/JSON 路径语法使用了一些 JavaScript 约定。

  • 点(.)用于成员访问。

  • 方括号([])用于数组访问。

  • SQL/JSON 数组是 0 相对的,与从 1 开始的常规 SQL 数组不同。

SQL/JSON 路径表达式中的数字文字遵循 JavaScript 规则,这些规则在某些细节上与 SQL 和 JSON 都不同。例如,SQL/JSON 路径允许 .11.,这在 JSON 中是无效的。支持非十进制整数文字和下划线分隔符,例如 1_000_0000x1EEE_FFFF0o2730b100101。在 SQL/JSON 路径(和 JavaScript 中,但在 SQL 本身中不是)中,不得在基数前缀之后直接使用下划线分隔符。

SQL/JSON 路径表达式通常在 SQL 查询中写为 SQL 字符串文字,因此它必须用单引号括起来,并且值中任何想要的单引号都必须加倍(参见 第 4.1.2.1 节)。某些路径表达式形式需要其中的字符串文字。这些嵌入的字符串文字遵循 JavaScript/ECMAScript 约定:它们必须用双引号括起来,并且可以在其中使用反斜杠转义来表示难以键入的字符。特别地,在嵌入式字符串文字中写入双引号的方法是 \",要写入反斜杠本身,您必须写入 \\。其他特殊反斜杠序列包括 JavaScript 字符串中识别的序列:\b\f\n\r\t\v 用于各种 ASCII 控制字符,\xNN 用于仅用两个十六进制数字编写的字符代码,\uNNNN 用于由其 4 位十六进制代码点标识的 Unicode 字符,以及 \u{N...} 用于用 1 到 6 位十六进制数字编写的 Unicode 字符代码点。

路径表达式由一系列路径元素组成,这些元素可以是以下任何一种:

  • JSON 原始类型的路径文字:Unicode 文本、数字、true、false 或 null。

  • 表 8.24 中列出的路径变量。

  • 表 8.25 中列出的访问器运算符。

  • jsonpath 运算符和方法列于 第 9.16.2.3 节

  • 括号,可用于提供过滤表达式或定义路径评估顺序。

有关使用 jsonpath 表达式与 SQL/JSON 查询函数的详细信息,请参阅 第 9.16.2 节

表 8.24. jsonpath 变量

变量 描述
$ 表示正在查询的 JSON 值(上下文项)的变量。
$varname 命名变量。它的值可以由几个 JSON 处理函数的参数vars设置;有关详细信息,请参阅 表 9.51
@ 表示过滤表达式中路径评估结果的变量。

表 8.25. jsonpath 访问器

访问器运算符 描述

.key

."$varname"

成员访问器,返回具有指定键的对象成员。如果键名与以 $ 开头的任何命名变量匹配,或者不符合 JavaScript 标识符规则,则必须用双引号括起来,使其成为字符串文字。

.*

通配符成员访问器,返回当前对象顶层所有成员的值。

.**

递归通配符成员访问器,处理当前对象 JSON 层次结构的所有级别,并返回所有成员的值,无论其嵌套级别如何。这是 SQL/JSON 标准的 PostgreSQL 扩展。

.**{level}

.**{start_levelend_level}

类似于 .**,但仅选择 JSON 层次结构的指定级别。嵌套级别用整数指定。级别零对应于当前对象。要访问最低嵌套级别,可以使用 last 关键字。这是 SQL/JSON 标准的 PostgreSQL 扩展。

[subscript, ...]

数组元素访问器。subscript 可以有两种形式:indexstart_indexend_index。第一种形式按索引返回单个数组元素。第二种形式通过索引范围返回一个数组切片,包括与提供的 start_indexend_index 对应的元素。

指定的 index 可以是整数,也可以是返回单个数值的表达式,该表达式会自动转换为整数。索引零对应于第一个数组元素。您还可以使用 last 关键字来表示最后一个数组元素,这对于处理未知长度的数组很有用。

[*]

通配符数组元素访问器,返回所有数组元素。




[7] 出于此目的,术语包括数组元素,尽管 JSON 术语有时将数组元素视为与对象内的值不同。

提交更正

如果您在文档中看到任何不正确的内容,与您对特定功能的体验不符,或者需要进一步澄清,请使用 此表单 报告文档问题。