MySQL JSON

MySQL5.7.7 labs版本开始在InnoDB存储引擎上提供原生JSON类型的支持,JSON值将不再以字符串的形式存储,而是采用一种允许快速读取文本元素(document elements)的内部二进制(internal binary)格式。在JSON列插入或更新的时候将自动验证JSON文本,未通过验证的文本将产生一个错误信息,JSON文本采用标准的创建方式,可使用大多数的比较操作符进行比较操作。

# 查看MySQL版本
$ mysql -V
mysql  Ver 14.14 Distrib 5.7.25, for Linux (x86_64) using  EditLine wrapper

# 查看MySQL版本
$ mysql --version
mysql  Ver 14.14 Distrib 5.7.25, for Linux (x86_64) using  EditLine wrapper
-- 查看MySQL版本
SELECT VERSION();
5.7.20-log

在此之前,我们通常会使用varchartext数据类型来存储JSON格式的数据。以前的方式是需要先从MySQL中读出来,然后在代码中修改后再存入,非常麻烦而且不科学。虽然MySQL有了原生的JSON数据类型后,但不知道性能如何呢?

优势特点

JSON类型与varchartext类型相比,有什么好处呢?

  • JSON数据类型的列会自动校验数据是否为JSON格式,若不是则会报错。
  • MySQL提供了一组操作JSON数据的内置函数
  • 存储在JSON列中的数据被转换成内部的存储格式,优化了存储格式,允许快速读取。
  • JSON数据有效性检查,Blob类型无法在数据库层做这样的约束性检查。
  • JSON数据查询是不需要遍历所有字符串,提升了查询性能。
  • JSON数据支持索引,可以通过虚拟列对JSON部分数据进行索引。

JSON数据格式

# 连接数据库
$ mysql -u username -p

# 查看所有数据库
$ show databases;

# 创建数据库
$ create database test;

# 选择数据库
$ use test;

如果不是十分熟悉数据库操作,可参见《MySQL命令》

-- 创建数据表
DROP TABLE IF EXISTS `game_config`;
CREATE TABLE IF NOT EXISTS `game_config`(
    `id` int(11) unsigned NOT NULL PRIMARY KEY AUTO_INCREMENT,
    `name` varchar(32) NOT NULL DEFAULT "",
    `title` varchar(32) NOT NULL DEFAULT "",
    `config` text DEFAULT NULL,
    `remark` varchar(255) NOT NULL DEFAULT ""
)ENGINE=Innodb DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;

-- 修改数据列的数据类型
ALTER TABLE `game_config` MODIFY `config` json;

-- 插入数据记录
INSERT INTO `game_config`(`name`,`title`,`config`) 
VALUE('version','版本','{"app_version":"1.0.0", "res_version":"1.0.0"}');

INSERT INTO `game_config`(`name`,`title`,`config`) 
VALUE('game','游戏','{"game_id":"10000", "game_type":1, "game_kind":2}');

INSERT INTO `game_config`(`name`,`title`,`config`) 
VALUE('login','登录','{"login_ip":"192.168.50.25", "login_port":9001, "api_url":"/api/game/login"}');

注意:

  • JSON列存储的必须是JSON格式数据,否则会报错。
  • JSON数据类型是没有默认值的

JOSN函数

MySQL中的JSON分为json arrayjson object两种类型,$表示整个json对象,在索引json array数据时使用下标,在索引json object数据时使用键值。

JSON_ARRAY(value1, value2, value3,...)

生成一个包含指定元素的JSON数组

mysql> SELECT JSON_ARRAY(1, "junchow", null, true, CURTIME());
+-------------------------------------------------+
| JSON_ARRAY(1, "junchow", null, true, CURTIME()) |
+-------------------------------------------------+
| [1, "junchow", null, true, "22:32:33.000000"]   |
+-------------------------------------------------+
1 row in set (0.12 sec)

JSON_OBJECT(key1, value1, key2, value2,...)

生成一个包含KV键值对的json object,如果keyNULL或参数个数为奇数则抛错。

mysql> SELECT JSON_OBJECT("id", 1, "name", "alice");
+---------------------------------------+
| JSON_OBJECT("id", 1, "name", "alice") |
+---------------------------------------+
| {"id": 1, "name": "alice"}            |
+---------------------------------------+
1 row in set (0.00 sec)

mysql> SELECT JSON_OBJECT("id", 1, "name", "alice", "nick");
ERROR 1582 (42000): Incorrect parameter count in the call to native function 'JSON_OBJECT'

mysql> SELECT JSON_OBJECT("id", 1, "name", "alice", null, "error");
ERROR 3158 (22032): JSON documents may not contain NULL member names.

CONVERT(json_string, JSON)

JSON字符串转换为JSON对象格式

mysql> SELECT CONVERT('{"id":1, "name":"junchow"}', JSON);
+---------------------------------------------+
| CONVERT('{"id":1, "name":"junchow"}', JSON) |
+---------------------------------------------+
| {"id": 1, "name": "junchow"}                |
+---------------------------------------------+
1 row in set (0.00 sec)

mysql> SELECT CONVERT('{id:1, name:"junchow"}', JSON);
ERROR 3141 (22032): Invalid JSON text in argument 1 to function cast_as_json: "Missing a name for object member." at position 1.

JSON_QUOTE(json_val)

JSON对象转换为JSON字符串,通过双引号字符包装并转义内部引号和其他字符,然后将结果作为utf8mb4字符串返回,并将字符串引用为JSON值。若参数为NULL则返回NULL

mysql> SELECT JSON_QUOTE('{id:1, name:"junchow"}');
+--------------------------------------+
| JSON_QUOTE('{id:1, name:"junchow"}') |
+--------------------------------------+
| "{id:1, name:\"junchow\"}"           |
+--------------------------------------+
1 row in set (0.00 sec)

mysql> SELECT JSON_QUOTE('NULL');
+--------------------+
| JSON_QUOTE('NULL') |
+--------------------+
| "NULL"             |
+--------------------+
1 row in set (0.00 sec)

JSON_CONTAINS(json_doc, val[, path])

查询json文档中是否在指定path上包含指定的数据,如果包含则返回1,否则返回0。如果有参数为NULL或者path不存在则返回NULL

# 从game_config表的config字段中判断game_kind为2的记录是否存在
mysql> SELECT JSON_CONTAINS(`config`, '2', '$.game_kind') FROM `game_config` WHERE `name`='game';

心得体会

其实我个人第一次看到有人用MySQL去存储JSON类型的数据时,是一头的火,非常窝火,因为我要取出来然后在代码中解析然后再这样倒腾来倒腾去,非常麻烦。因为有NoSQL,存JSON干嘛不直接用MongoDB。不过有了原生的JSON函数支持,也逐渐开始学习适应。避开好恶,RDB和NoSQL应用场景是不同的,相煎何太急。同事很喜欢用MySQL去存配置,到我这里做数据统计时,又是一头的抓狂。哎,为了省事,其实很多时候并不省事儿。既然使用到RDB,具有关系属性的字段是可以提取出来的,变化的字段和简单的配置也是可以使用的。文章的内容也是用到学到,学到用到,慢慢增加积累,一味的为了知识而学习,是一种不良作风,实践是需要场景而非模拟,才能体会到它的价值所在。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,997评论 6 502
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,603评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,359评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,309评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,346评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,258评论 1 300
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,122评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,970评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,403评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,596评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,769评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,464评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,075评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,705评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,848评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,831评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,678评论 2 354