PostgreSQL 务实应用(四/5)JSON
JSON 可谓风靡互联网,在数据交换使用上,其优势特别明显,其结构简洁、可读易读、形式灵活。很多 API 接口的数据都采用 JSON 来表示。
PostgreSQL 对 JSON 提供了良好的支持。具体的相关函数可参考:JSON类型和函数
从使用的角度而言,个人觉得常见的应用场景为:
- 读取单个 JSON 的属性值
- 遍历单个 JSON 的所有属性
- 遍历一个 JSON 数组
- 创建一个 JSON 作为返回值
之所以仅这些简单的场景,原因在于,在应用中使用高级语言处理 JSON 与在数据库中使用那些高级的 JSON 函数相比,从操作上和可读性上均爽很多。在不支持 JSON 的数据库中,我们也常使用单个文本字段存储 JSON 字符串,然后在应用中加以解析处理。
四个场景
我们以以下的 JSON 字符串作为输入,来了解 PostgreSQL 在各场景中的应用实现。
{
"label": {
"names": ["Amy", "Kala", "Lily"]
},
"color": "red",
"count": 3
}
// 写成一行即是
{"label":{"names":["Amy","Kala","Lily"]},"color":"red","count":3}
1.读取属性
首先,通过下面的表格,感受一下 JavaScript 与 PostgreSQL 中读取 color 属性与 label 属性中 names 的第二个值的形式。
读属性 | JavaScript | PostgreSQL |
---|---|---|
定义 | var jsonObj = {"label":{"names":["Amy","Kala","Lily"]}, "color":"red","count":3}; | jsonObj := '{"label":{"names":["Amy","Kala","Lily"]}, "color":"red","count":3}'::json; |
读取 JSON 的 color 属性 | jsonObj.color | jsonObj -> 'color' |
读取 JSON 的 label 中 names 的第二个值 | jsonObj.label.names[1] | jsonObj -> 'label' -> 'names' -> 1 |
在 PostgreSQL 中我们可以使用以下语句逐层指定属性路径(属性名称需要使用字符串需单引号,数组索引使用数字)来获取值:
-- 取得 color 属性
SELECT '{"label":{"names":["Amy","Kala","Lily"]},"color":"red","count":3}'::json
-> 'color';
-- 取得 label 属性下的 names 的第二个值
SELECT '{"label":{"names":["Amy","Kala","Lily"]},"color":"red","count":3}'::json
-> 'label' -> 'names' -> 1;
此时取得的值仍然为 json 类型,如果需要取得值的文本形式,则把最后一个 "->" 变成 "->>" 即可。
当然,路径的表示,也可以通过 #> '{label,names,1}' 的形式表示:
SELECT '{"label":{"names":["Amy","Kala","Lily"]},"color":"red","count":3}'::json
#> '{label,names,1}';
2.遍历属性
使用 json_each 函数,即可返回属性键值对的数据集,数据集包括两列,key 表示属性,value 表示属性值。如下语句输出所有结果:
DO $$
DECLARE
lv_row record;
jsonObj json := '{"label":{"names":["Amy","Kala","Lily"]},"color":"red","count":3}'::json;
BEGIN
FOR lv_row IN SELECT * FROM json_each(jsonObj) LOOP
raise notice 'key is %, value is %', lv_row.key, lv_row.value;
END LOOP;
END $$;
输出
NOTICE: key is label, value is {"names":["Amy","Kala","Lily"]}
NOTICE: key is color, value is "red"
NOTICE: key is count, value is 3
3.遍历数组
通过使用 json_array_length 函数获取数组的长度,然后根据索引遍历整个数组即可。
DO $$
DECLARE
lv_row record;
lv_size int;
jsonObj json := '{"label":{"names":["Amy","Kala","Lily"]},"color":"red","count":3}'::json;
BEGIN
-- 取得label 下names 这个json数组
jsonObj := jsonObj #> '{label,names}';
-- 取得数组的长度
lv_size := json_array_length(jsonObj);
-- 按索引遍历整个数组
FOR i IN 0..lv_size-1 LOOP
raise notice '%', jsonObj -> i;
END LOOP;
END $$;
输出:
NOTICE: "Amy"
NOTICE: "Kala"
NOTICE: "Lily"
4.创建一个 JSON
使用 json_build_object 函数,传递 key, value 成对的参数即可创建一个 json,如以下语句形成一个 api 常用的返回执行情况的 json。
SELECT json_build_object('code', 200, 'err_msg', 'run success!');
应用示例
我们以填写学生地址为例,传递给存储过程的是一个 json 数组,每个数组中的 json 对象包括了学生标识与地址信息。
以下语句创建数据表
-- student_id 学生标识, address 地址
CREATE TABLE student_address (student_id varchar(10) PRIMARY KEY, address varchar(100));
以下为处理过程
CREATE OR REPLACE FUNCTION save_student_addresses_json(
v_array_json json)
RETURNS json
LANGUAGE 'plpgsql'
AS $$
DECLARE
lv_row_json json;
lv_length int;
lv_field_student_id varchar;
lv_field_address varchar;
BEGIN
-- 取得数组的长度
lv_length := json_array_length(v_array_json);
FOR i IN 0..lv_length-1 LOOP
-- 取得第 i 行的 json 值
lv_row_json := v_array_json -> i;
lv_field_student_id := lv_row_json ->> 'student_id';
lv_field_address := lv_row_json ->> 'address';
-- 插入学生地址信息,如果存在则更新地址
INSERT INTO student_address (student_id, address)
VALUES (lv_field_student_id, lv_field_address)
ON CONFLICT (student_id)
DO UPDATE SET address = excluded.address;
END LOOP;
RETURN json_build_object(
'err_code', 200,
'err_msg', '保存或更新 ' || lv_length || ' 条记录'
);
end
$$
我们执行以下操作
SELECT save_student_addresses_json(
'[
{"student_id":"01","address":"街道A"},
{"student_id":"02","address":"街道B"}
]'
);
运行结果:{"err_code":200,"err_msg":"保存或更新 2 条记录"}
小结一下
PostgreSQL 对 JSON 的操作支持特性很丰富,但文档中那么多函数一下映入眼帘,让人觉得复杂凌乱。本文从简单易理解的几个应用场景出发,希望能先爽上一把,而后再细细深入。I love PostgreSQL!
PostgreSQL 务实应用(四/5)JSON的更多相关文章
- {Django基础七之Ajax} 一 Ajax简介 二 Ajax使用 三 Ajax请求设置csrf_token 四 关于json 五 补充一个SweetAlert插件(了解)
Django基础七之Ajax 本节目录 一 Ajax简介 二 Ajax使用 三 Ajax请求设置csrf_token 四 关于json 五 补充一个SweetAlert插件(了解) 一 Ajax简介 ...
- JAVA中的四种JSON解析方式详解
JAVA中的四种JSON解析方式详解 我们在日常开发中少不了和JSON数据打交道,那么我们来看看JAVA中常用的JSON解析方式. 1.JSON官方 脱离框架使用 2.GSON 3.FastJSON ...
- python学习第四十八天json模块与pickle模块差异
在开发过程中,字符串和python数据类型进行转换,下面比较python学习第四十八天json模块与pickle模块差异. json 的优点和缺点 优点 跨语言,体积小 缺点 只能支持 int st ...
- {Django基础七之Ajax} 一 Ajax简介 二 Ajax使用 三 Ajax请求设置csrf_token 四 关于json 五 补充一个SweetAlert插件(了解)
{Django基础七之Ajax} 一 Ajax简介 二 Ajax使用 三 Ajax请求设置csrf_token 四 关于json 五 补充一个SweetAlert插件(了解) Django基础七之 ...
- PostgreSQL 务实应用(五/5)常用表达
在实际应用中,对于具体的数据计算我们会找相应的函数来实现.而计算需求不同的表达,往往会使得我们使用不同的函数或方式来实现.或者也可以说,同一计算可以使用多种不同的表达方式实现. PostgreSQL ...
- python接口自动化四(json数据处理)
前言 有些post的请求参数是json格式的,这个前面第二篇post请求里面提到过,需要导入json模块处理. 一般常见的接口返回数据也是json格式的,我们在做判断时候,往往只需要提取其中几个关键的 ...
- python3.x 基础四:json与pickple
每次打开一个文件,只dump1次 json.dump(dump的内容,文件句柄) json.load(文件句柄) json可以处理列表/字典/字符串等简单数据类型,但是不能处理复杂的数据类型,如函数的 ...
- python基础四(json\os\sys\random\string模块、文件、函数)
一.文件的修改 文件修改的两种思路: 1.把文件内容拿出来,做修改后,清空原来文件的内容,然后把修改过的文件内容重新写进去. 步骤: 1.打开文件:f=open('file','a+') #必须用a ...
- (十四)json、pickle与shelve模块
任何语言,都有自己的数据类型,那么不同的语言怎么找到一个通用的标准? 比如,后端用Python写的,前端是js,那么后端如果传一个dic字典给前端,前端肯定不认. 所以就有了序列化这个概念. 什么是序 ...
随机推荐
- mysql 数据库使用
1. 常用命令 2. 创建表 create table test ( test_id int, test_price decimal ); 或者 create table test2 as selec ...
- 大话设计模式--桥接模式 Bridge -- C++实现实例
1. 桥接模式: 将抽象部分与它的实现部分分离,使它们都可以独立的变化. 分离是指 抽象类和它的派生类用来实现自己的对象分离. 实现系统可以有多角度分类,每一种分类都有可能变化,那么把这种多角度分离出 ...
- Spring Boot- 用idea新建spring boot web项目
1.新建project 2.选择Spring Initializr,next 3.输入项目信息,next 4.选择web依赖以及Spring Boot的版本,next 5.Finish 6.Enabl ...
- OpenCV——饱和度调整
参考: 闲人阿发伯的博客 // define head function #ifndef PS_ALGORITHM_H_INCLUDED #define PS_ALGORITHM_H_INCLUDED ...
- Ffmpeg移植S3C2440
Ffmpeg移植过程: FFmpeg是一个开源免费跨平台的视频和音频流方案,属于自由软件,采用LGPL或GPL许可证.它的移植同样遵循LGPL或GPL移植方法:configure.make.make ...
- ffmpeg处理RTMP流媒体的命令和发送流媒体的命令(UDP,RTP,RTMP)
将文件当做直播送至live ffmpeg -re -i localFile.mp4 -c copy -f flv rtmp://server/live/streamName re限制输出速率, ...
- LeetCode Majority Element I
原题链接在这里:https://leetcode.com/problems/majority-element/ 题目: Given an array of size n, find the major ...
- iOS使用NSURLSession发送POST请求,后台无法接受到请求过来的参数
iOS中发送POST请求,有时需要设置Content-Type,尤其是上传图片的时候. application/x-www-form-urlencoded: 窗体数据被编码为名称/值对.这是标准的编码 ...
- nodejs调试:node-inspector
基于Chrome浏览器的调试器 既然我们可以通过V8的调试插件来调试,那是否也可以借用Chrome浏览器的JavaScript调试器来调试呢?node-inspector模块提供了这样一种可能.我们需 ...
- Trilead,SSH2的Java调用
最近项目要部署10台设备,如果每台设备都手动进行部署想想也是醉了. 因为之前一直使用SecurityFX以及SecurityCRT,所以考虑是否可以使用基于SSH2的类库来实现文件拷贝以及远程命令调用 ...