mySQL中的JSON操作

关于JSON值

  • json数组可以包含数字、字符、布尔值等,以[]组合起来
[10,null,"a",true]
  • json对象的健值表现形式以键值对表示,且值可以是字符型或数字、json空文本或json布尔值,但键必须是字符型,以{}组合起来
{"key1":"value1","key2":8}
  • json还允许使用时间(date``time``datetime``timestamp)等
["15:51:29.000000", "2019-03-20", "2019-043-20 15:51:29.000000"]
  • 在json数组和对象中允许嵌套
["ac",{"key1":"value1"}]
{"key2":10,[null,true]}

mySQL中的创建JSON值的函数

JSON_ARRAY([value[,value]…)
  • 创建json数组,并返回该数组

  • 可以为null

SELECT JSON_ARRAY(9,null,true,"I am Dolores",now());
//output
[9, null, true, "I am Dolores", "2019-03-20 16:05:42.000000"]
JSON_ARRAYAGG(value)

创建单个json数组

 SELECT JSON_ARRAYAGG(1);//[1]
 SELECT JSON_ARRAYAGG(null);//[null]
 SELECT JSON_ARRAYAGG(true);//[true]
 SELECT JSON_ARRAYAGG(now());//["2019-03-20 16:31:20.000000"]
JSON_OBJECT([key,value[,key,value]…)
  • 创建json对象,并返回该键值对

  • 如果任何键名称为null或参数个数为奇数,则发生错误

SELECT json_object('name','Dolores','id','2015214328');
//output
{"id": "2015214328", "name": "Dolores"}
JSON_OBJECTAGG(key,value)

创建单个对象,并返回该对象

SELECT json_objectagg('ID',22);
//{"ID": 22}
JSON_QUOTE(string)
  • 将字符串作为json值引用,方法是用双引号字符包装字符串,并转义内部引号和其他字符,然后将结果以utf8bm4字符串返回

  • 如果参数为null,则返回null

SELECT JSON_QUOTE('null');
//output
"null"

SELECT JSON_QUOTE('"null"');
//output
"\"null\""

SELECT JSON_QUOTE('I \'m Dolores');
//output
"I 'm Dolores"

SELECT JSON_QUOTE('[1,2,3]');
//output
"[1,2,3]"
CAST(value AS JSON)

将其他类型的值转换成JSON类型来获取json值

SELECT CAST('null' AS JSON)
//"null"

判断JSON

JSON_TYPE()

json值作为参数传入,如果值有效,则返回其json类型,否则报错

SELECT JSON_TYPE('[1,"Dolores",true]');
//ARRAY

SELECT JSON_TYPE('"hello"');
//STRING

SELECT JSON_TYPE('{"key":"value"}');
//OBJECT

SELECT JSON_TYPE('hello');
//Error Code: 3141. Invalid JSON text in argument 1 to function json_type: "Invalid value." at position 0.  0.000 sec

合并JSON

JSON_MERAGE_PRESERVE()

将两个或多个json值合并为一个json并返回最终值

SELECT JSON_MERGE_PRESERVE('["ABC","HELLO"]','{"name":"Dolores"}');
//["ABC", "HELLO", {"name": "Dolores"}]
JSON_MERGE_PATCH()

合并两个或多个json值,但不合并重复键的值,如果出现重复键,仅保留最后一个的值

SELECT JSON_MERGE_PATCH('[1, 2]', '[true, false]');
//[true, false]     

SELECT JSON_MERGE_PATCH('{"name": "x"}', '{"id": 47}');
//{"id": 47, "name": "x"}

SELECT JSON_MERGE_PATCH('{"a":{"x":1}}', '{"a":{"y":2}}');
// {"a": {"x": 1, "y": 2}}

两者的比较
  • 在遇到重复键时,json_merge_patch仅保留最后一个值,

  • json_merge_perserve会在原有值上添加值

SET @x = '{ "a": 1, "b": 2 }',
@y = '{ "a": 3, "c": 4 }',
@z = '{ "a": 5, "d": 6 }';

SELECT  JSON_MERGE_PATCH(@x, @y, @z)  AS `Patch`,
JSON_MERGE_PRESERVE(@x, @y, @z) AS `Preserve`
//Patch: {"a": 5, "b": 2, "c": 4, "d": 6}
//Preserve: {"a": [1, 3, 5], "b": 2, "c": 4, "d": 6}
mysql中的json是否区分大小写

经过函数转换得到的json是区分大小写的,原因在于转换后的字符集格式为utf8mb4utf8mb4_bin,因为utf8mb4_bin是二进制排序规则,所以区分大小写

SET @j = JSON_OBJECT('key', 'value');
SELECT CHARSET(@j), COLLATION(@j);
//utf8mb4     utf8mb4_bin   
SELECT JSON_ARRAY('a') = JSON_ARRAY('A');
//0
(0为false的意思)

因为区分大小写,所以json中的nulltruefalse都必须用小写字母编写

SELECT JSON_VALID('null'), JSON_VALID('Null'), JSON_VALID('NULL');
//1 0 0 

SELECT JSON_VALID('true'), JSON_VALID('True'), JSON_VALID('TRUE');
/1 0 0 

需要存储转义字符时

直接插入键值对语句和用JSON_OBJECT转换成json值存入的差别在于,前者需要双反斜杠转义字符,而后者只需要单反斜杠转义字符

CREATE TABLE facts (sentence JSON);

当需要存储的内容如下

mascot: The MySQL mascot is a dolphin named "Sakila".

使用直接插入的方法时:

INSERT INTO facts VALUES
('{"mascot": "Our mascot is a dolphin named \\"Sakila\\"."}');

SELECT sentence FROM facts;
// {"mascot": "Our mascot is a dolphin named \"Sakila\"."}

使用JSON_OBJECT

 INSERT INTO facts VALUES
(JSON_OBJECT("mascot", "Our mascot is a dolphin named \"Sakila\"."));

SELECT sentence FROM facts;
//{"mascot": "Our mascot is a dolphin named \"Sakila\"."}

搜索和修改json---json路径表达式

json路径表达式用于精确查找json中的值

  • 路径语法使用前导$来表示正在考虑的 json 文档,后跟选择器

  • 选择器的语法为:

    • .后跟一个具体的键名,如'$.name',当有空格是必须加上双引号如'$."my name"'
     SELECT JSON_EXTRACT('{"id": 14, "name": "Aztalan"}', '$.name');
     //"Aztalan" ​    
    
    • 使用[N], 该路径命名数组中位置 N 的值。数组位置是以零开头的整数。如果path未选择数组值, path 的计算结果与path[0]相同:
    //JSON_SET是一个具备修改功能的函数
    SELECT JSON_SET('"x"', '$[0]', 'a');
     //"a"
    SELECT JSON_SET('"x"', '$[1]', 'a','$[2]','B');
    //["x", "a", "B"]
    
    • [M to N]指定数组值的子集或范围, 从位置 M的值开始, 并以位置 N 的值结束.

      SELECT JSON_EXTRACT('[1, 2, 3, 4, 5]', '$[1 to 3]');
      //[2, 3, 4]    
      
      • 可以配合last使用

         SELECT JSON_EXTRACT('[1, 2, 3, 4, 5]', '$[last-3 to last-1]');
        //[2, 3, 4]  
        
         SELECT JSON_REPLACE('"Sakila"', '$[last]', 10);
         //10
        
  • 路径可以包含***通配符:

  • .*计算为JSON 对象中所有成员的值。

    ```mysql
    SELECT JSON_EXTRACT('{"a": 1, "b": 2, "c": [3, 4, 5]}', '$.*');
    //[1, 2, [3, 4, 5]]  
    ```
    
  • [*]计算为JSON数组中所有元素的值。

    ```mysql
     SELECT JSON_EXTRACT('{"a": 1, "b": 2, "c": [3, 4, 5]}', '$.c[*]');
     //[3, 4, 5] 
    ```
    
    • prefix**suffix计算结果为以命名前缀开头并以命名后缀结尾的所有路径。

      如路径$**.b计算为多个路径 ($.a.b . b 和$.c.b), 并生成匹配路径值的数组

       SELECT JSON_EXTRACT('{"a": {"b": 1}, "c": {"b": 2}}', '$**.b');
       // [1, 2] 
      
  • 文档中不存在的路径 (计算结果为不存在的数据) 的计算结果为NULL.

案例

[3, {"a": [5, 6], "b": 10}, [99, 100]]
  • $[0]的计算结果3.
  • $[1]的计算结果为{"a": [5, 6], "b": 10}.
  • $[2]的计算结果为[99, 100].
  • $[3]的计算结果为NULL (它指的是第四个数组元素, 它不存在)。

因为$[1]$[2]计算为非标量值, 所以它们可以用作选择嵌套值的更具体的路径表达式的基础。例子:

  • $[1].a的计算结果为[5, 6].
  • $[1].a[1]的计算结果6.
  • $[1].b计算结果10.
  • $[2][0]结果为99.
{"a fish": "shark", "a bird": "sparrow"}
  • $."a fish"的结果是shark.
  • $."a bird"的结果是sparrow.

结合JSON_SET``JSON_INSERT``JSON_REPLACE``JSON_REMOVE的使用

SET @j = '["a", {"b": [true, false]}, [10, 20]]';

JSON_SET替换存在的路径的值, 并为不存在的路径添加值

SELECT JSON_SET(@j, '$[1].b[0]', 1, '$[2][2]', 2);
// ["a", {"b": [1, false]}, [10, 20, 2]]

JSON_INSERT添加新值, 但不替换现有值:

SELECT JSON_INSERT(@j, '$[1].b[0]', 1, '$[2][2]', 2);
// ["a", {"b": [true, false]}, [10, 20, 2]] 

JSON_REPLACE替换现有值并忽略新值:

 SELECT JSON_REPLACE(@j, '$[1].b[0]', 1, '$[2][2]', 2);
 // ["a", {"b": [1, false]}, [10, 20]]  

JSON_REMOVE使用一个或多个路径, 这些路径指定要从文档中删除的值。返回值是原始文档减去由文档中存在的路径选择的值:

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

推荐阅读更多精彩内容