百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程字典 > 正文

PostgreSQL是不是你的下一个JSON数据库

toyiye 2024-06-21 12:24 11 浏览 0 评论

根据Betteridge定律(任何头条的设问句可以用一个词来回答:不是),除非你的JSON数据很少修改,并且查询很多。

最新版的PostgreSQL添加更多对JSON的支持,我们曾经问过PostgreSQL是否可以替换MongoDB作为JSON数据库,答案显而易见,但我们更希望的是,啊哈,这个问题由读者来问了。

“PostgreSQL不是已经有一些json的支持了吗?”

是的,在PostgreSQL 9.4之前的版本也有JSON 数据类型了,你可以这样:

CREATE TABLE justjson ( id INTEGER, doc JSON)
>INSERT INTO justjson VALUES ( 1, '{
    "name":"fred",
    "address":{
        "line1":"52 The Elms",
        "line2":"Elmstreet",
        "postcode":"ES1 1ES"
        }
    }');

保存了JSON的原始文本到数据库,包括空白行和键顺序及重新的键,我们来查看下保存的数据:

>SELECT * FROM justjson;
 id |               doc
----+---------------------------------
  1 | {                              +
    |     "name":"fred",             +
    |     "address":{                +
    |         "line1":"52 The Elms", +
    |         "line2":"Elmstreet",   +
    |         "postcode":"ES1 1ES"   +
    |         }                      +
    |     }
(1 row)

跟保存之前的文本一模一样,但我们仍可以解析出具体的数据出来,PostgreSQL提供了一套JSON的操作方法进行查找,例如,我们只要查出address信息,如果做?

select doc->>'address' FROM justjson;  
            ?column?
---------------------------------
 {                              +
         "line1":"52 The Elms", +
         "line2":"Elmstreet",   +
         "postcode":"ES1 1ES"   +
         }
(1 row)

doc字段的 ->> 操作符是查询JSON对象的某个字段并返回文本,用数字也可以当作数组的索引,但仍返回文本。跟 ->> 类似的还有 -> 操作符,返回不转文本的内容,可以用它来导航搜索JSON对象,如:

select doc->'address'->>'postcode' FROM justjson;  
 ?column?
----------
 ES1 1ES
(1 row)

还有个更简短的写法来指定搜索路径,用 #>> 操作符,如梦:

select doc#>>'{address,postcode}' FROM justjson;  
 ?column?
----------
 ES1 1ES
(1 row)

通过保存完整的JSON数据类型可使其跟源数据完全一样并且不会丢失内容,但为保持完全一致也带来了成本,性能的缺失,而且不能索引...所有,尽管可以很方便的维持一致性和保持JSON文档,但仍有很大的提升空间,所以引入了JSONB。

"JSONB有什么不同?"

JSONB可以将整个JSON文档转有层级的KEY/VALUE数据对,所有的空白字符删除了,重复键只保留最后一次,键也没有排序,而是用HASH来保存了,上面的例子中用JSONB的版本的话,看来起类似这样:

>CREATE TABLE justjsonb ( id INTEGER, doc JSONB)
>INSERT INTO justjsonb VALUES ( 1, '{
    "name":"fred",
    "address":{
        "line1":"52 The Elms",
        "line2":"Elmstreet",
        "postcode":"ES1 1ES"
        }
    }');
>SELECT * FROM justjsonb;
 id |                                                doc
----+----------------------------------------------------------------------------------------------------
  1 | {"name": "fred", "address": {"line1": "52 The Elms", "line2": "Elmstreet", "postcode": "ES1 1ES"}}
(1 row)

可以看到,所有非文本内容都消失了,替换成JSON文档需要的最少格式,这种压缩方式表示当数据插入时会自动格式化,这样可以减少之后访问数据分析处理的工作量。

"PostgreSQL的这种数据有点像HSTORE"

看到键值对,JSONB还真有点像PostgreSQL的HSTORE扩展,它也可以保存键值对,但它是一个扩展,而,JSONB(以及JSON)是在PostgreSQL内核的,HSTORE只有一级层级,但PostgreSQL可以有嵌套的元素,并且,HSTORE只能存字符串,而JSONB还可以存JSON的所数字类型。

“那JSONB到底带给我啥好处呢?”

索引,到处用上索引,你不能在PostgreSQL对JSON类型创建真正的索引,你可以创建表达式索引(expression indexes),但只限于你想索引的内容,例如:

create index justjson_postcode on justjson ((doc->'address'->>'postcode'));  

只有邮编(postcode)索引了,其它都没有索引。

而JSONB,支持GIN索引,一种通用返转索引(Generalized Inverted Index),PostgreSQL提供了另外一套索引操作符来支持,包括 @> 包括JSON,<@ 最包含,? 测试字符串是否存在,?| 任意字符串是否存在,?& 所有存大的字符串。

有两类索引可用,默认叫 json_ops,它支持所有操作符(译者:指普通json操作符)和一个只支持&>操作符的jsonb_path_ops索引(译者:指索引操作符),默认索引给JSON中的每个键值都创建了索引,其实 jsonb_path_ops只创建了一个比默认复杂的更高压缩的hash表索引,但默认索引担任更多操作能力同时增加了空间成本。给表添加一些数据,我们再来看看某个邮编,如果我们创建了一个默认的GIN JSON索引然后查询:

explain select * from justjsonb where doc @> '{ "address": { "postcode":"HA36CC" } }';  
                           QUERY PLAN
-----------------------------------------------------------------
 Seq Scan on justjsonb  (cost=0.00..3171.14 rows=100 {"address": {"postcode": "HA36CC"}}'::jsonb)
(2 rows)

可以看出来是顺序扫瞄表,如果我们加个默认的JSON GIN索引后再看看有什么不同?

> create index justjsonb_gin on justjsonb using gin (doc);
> explain select * from justjsonb where doc @> '{ "address": { "postcode":"HA36CC" } }';
                                  QUERY PLAN
-------------------------------------------------------------------------------
 Bitmap Heap Scan on justjsonb  (cost=40.78..367.62 rows=100 {"address": {"postcode": "HA36CC"}}'::jsonb)
   ->  Bitmap Index Scan on justjsonb_gin  (cost=0.00..40.75 rows=100 {"address": {"postcode": "HA36CC"}}'::jsonb)
(4 rows)

搜索性能提升很大,但隐藏了空间的耗费,例中是41%的数据大小,让我们删除索引重复执行jsonb_path_ops GIN索引。

> create index justjsonb_gin on justjsonb using gin (doc jsonb_path_ops);
> explain select * from justjsonb where doc @> '{ "address": { "postcode":"HA36CC" } }';
                                  QUERY PLAN
-------------------------------------------------------------------------------
 Bitmap Heap Scan on justjsonb  (cost=16.78..343.62 rows=100 {"address": {"postcode": "HA36CC"}}'::jsonb)
   ->  Bitmap Index Scan on justjsonb_gin  (cost=0.00..16.75 rows=100 {"address": {"postcode": "HA36CC"}}'::jsonb)
(4 rows)

总成本低了点,索引体积小了很多,这是典型的创建索引速度和空间平衡的方法,但比顺序扫瞄性能高很多。

“我应该用它作为我的JSON数据库吗?”

如果你经常更新你的JSON文档,回答是否定的,PostgreSQL最擅长的是存储和攻取JSON文档及他们的字段,但尽管如此你可以取出单个字段,你也不能更新单个字段;实际上你可以,将整个JSON解析出来,添加新的字段再写回,让JSON分析器处理重复,但你很明显不想依赖这个。

如果你的主要数据用关系数据库用得很好,JSON数据只是一群补充(静态数据),那么用PostgreSQL就可以了,而且用JSONB表示和索引能力将更高效。另外,如果你的数据模型是可变内容的集合,那么你可能会寻找一样主流工业级的json文档数据库如MongoDB或RethinkDB。

相关推荐

电子书 | 笨办法学 Python 3(笨办法学python3pdf)

本周更新了5本IT电子书资源,同时站内已经有12本Python入门方面的相关电子书,可供新手选择。1、笨办法学Python3本书是一本Python入门书,适合对计算机了解不多,没有...

Python2 已终结,入手Python 3,你需要这30个技巧

选自medium作者:Erik-JanvanBaaren机器之心编译参与:王子嘉、一鸣Python2在今年和我们说拜拜了,Python3有哪些有趣而又实用的技巧呢?这篇教程有30个你会喜欢...

Python 3 系列教程(python3.9基础教程)

Python的3.0版本,常被称为Python3000,或简称Py3k。相对于Python的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python3.0在设计的时候没有考...

Python第三课3. Python 的非正式介绍

3.Python的非正式介绍?在下面的例子中,通过提示符(>>>与...)的出现与否来区分输入和输出:如果你想复现这些例子,当提示符出现后,你必须在提示符后键入例子中的每...

如何使用 Python 构建一个“谷歌搜索”系统?| 内附代码

来源|hackernoon编译|武明利,责编|Carol出品|AI科技大本营(ID:rgznai100)在这篇文章中,我将向您展示如何使用Python构建自己的答案查找系统。基本上,这...

Python 模拟微博登陆,亲测有效!(如何用python爬微博)

今天想做一个微博爬个人页面的工具,满足一些不可告人的秘密。那么首先就要做那件必做之事!模拟登陆……代码是参考了:https://www.douban.com/note/201767245/,我对代码进...

Python 驱动的 AI 艺术批量创作: 免费的Bing 绘图代码解析

这篇文章将深入分析一段Python代码,该代码利用Bing的AI绘图功能,即bing的images/create,根据用户提供的文本提示生成图像。我们将详细探讨其工作原理、代码结构、...

Python爬虫Scrapy库的使用入门?(python scrapy爬虫)

Scrapy是一个开源的并且支持高度可扩展的Python爬虫框架,主要被用来实现从网站提取数据。出现之初就是为网页抓取而设计,但是现在它也可以被用于从APIs中抓取数据或通用的Web抓取任务。Sc...

Python3 标准库概览(python标准库有什么)

操作系统接口os模块提供了不少与操作系统相关联的函数。>>>importos>>>os.getcwd()#返回当前的工作目录'C:\\Python34...

零基础入门学习Python(三):变量和字符串

分享兴趣,传播快乐,增长见闻,留下美好!亲爱的您,这里是LearningYard新学苑。今天小编为大家带来的是...

Python读写docx文件(python读写word)

Python读写docx文件Python读写word文档有现成的库可以处理pipinstallpython-docx安装一下。https://python-docx.readthedocs.io/...

如何利用Xpath抓取京东网商品信息

前几小编分别利用Python正则表达式和BeautifulSoup爬取了京东网商品信息,今天小编利用Xpath来为大家演示一下如何实现京东商品信息的精准匹配~~HTML文件其实就是由一组尖括号构成的标...

如何利用Xpath选择器抓取京东网商品信息

前几小编分别利用Python正则表达式和BeautifulSoup爬取了京东网商品信息,今天小编利用Xpath来为大家演示一下如何实现京东商品信息的精准匹配~~HTML文件其实就是由一组尖括号构成的标...

python之Scrapy爬虫案例:豆瓣(python爬虫书籍豆瓣评分)

python模块之Scrapy爬虫框架...

Python编程入门学习:最常见加密方式和Python实现

前言我们所说的加密方式,都是对二进制编码的格式进行加密的,对应到Python中,则是我们的Bytes。所以当我们在Python中进行加密操作的时候,要确保我们操作的是Bytes,否则就会报错。将字符串...

取消回复欢迎 发表评论:

请填写验证码