首頁 > 軟體

MySQL之JSON型別欄位的使用技巧分享

2022-08-25 18:03:49

測試環境: MySQL8.0.19

準備工作

CREATE TABLE json_demo ( 
	`id` INT ( 11 ) NOT NULL PRIMARY KEY, 
	`content` json NOT NULL 
);
INSERT INTO json_demo ( id, content )
VALUES
	/*這條是陣列*/
	( 1, '[{"key": 1, "order": 1, "value": "34252"},{"key": 2, "order": 2, "value": "23423"}]' ),
	/*這條是陣列*/
	( 2, '[{"key": 4, "order": 4, "value": "234"},{"key": 5, "order": 5, "value": "234324523"}]' ),
	/*這條是物件*/
	( 3, '{"key": 3, "order": 3, "value": "43242"}' ),
	/*這條是物件*/
	( 4, '{"key": 6, "order": 6, "value": "5423"}' );

JSON物件基礎操作

查詢指定欄位值

/* 基礎查詢 */
SELECT
	content -> '$.key' AS 'key',
	JSON_EXTRACT(content, '$.key') AS 'key2',
	content -> '$.value' AS 'value',
	JSON_EXTRACT(content, '$.value') AS 'value2',
	content ->> '$.value' AS 'value3',
	JSON_UNQUOTE(JSON_EXTRACT(content, '$.value')) AS 'value4'
FROM
	json_demo 
WHERE
	id > 2;

TIPS:

  • ->和->>是MySQL設計的語法,其中->在MySQL5.7支援,->>在MySQL8.0中支援。
  • ->等效於JSON_EXTRACT(),當查詢欄位為字串時,其返回值還會帶有""。
  • ->>等效於JSON_UNQUOTE(JSON_EXTRACT()),當查詢欄位為字串時,其返回值不會帶有""。

用於條件查詢

content -> '$.key'可以看成一個欄位,一個欄位能做的操作基本他都能。

SELECT
	id,
	content -> '$.key' AS 'key',
	content ->> '$.value' AS 'value3'
FROM
	json_demo 
WHERE
	id > 2
	AND content -> '$.key' > 1
	AND content -> '$.value' like '%2%';

修改指定欄位值

/* 修改 */
UPDATE json_demo 
SET content = JSON_REPLACE(
	content,
	/* 將content.key值 + 1 */
	'$.key', content -> '$.key' + 1,
	/* 將content.value值後拼接'abc' */
	'$.value', concat(content ->> '$.value', 'abc')
) WHERE id = 3;
/* JSON_SET也可以 */
UPDATE json_demo 
SET content = JSON_SET(
	content,
	/* 將content.key值 + 1 */
	'$.key', content -> '$.key' + 1,
	/* 將content.value值後拼接'abc' */
	'$.value', concat(content ->> '$.value', 'abc')
) WHERE id = 3;
/* 查詢修改結果 */
SELECT id,content,content -> '$.key' AS 'key',content ->> '$.value' AS 'value3'
FROM json_demo WHERE id = 3;
/* 重新賦值 */
UPDATE json_demo SET 
content = JSON_REPLACE(content,'$.key',3,'$.value','43242') WHERE id = 3;

TIPS:

  • JSON_REPLACE和JSON_SET都可以用來修改某個欄位值,區別在於JSON_REPLACE替換不存在的屬性時操作無效;而JSON_SET則會將這個不存在的屬性插入進去。
  • 所以JSON_SET也可以用來追加屬性,與JSON_INSERT類似。區別在於JSON_INSERT如果插入一個已存在的屬性時操作會失效,而JSON_SET會替換。

追加元素

UPDATE json_demo 
SET content = JSON_INSERT(content, '$.key', 234)
WHERE id = 3;

SELECT id,content,content -> '$.key' AS 'key' FROM json_demo WHERE id = 3;

UPDATE json_demo 
SET content = JSON_INSERT(content, '$.temp', 234)
WHERE id = 3;

SELECT id,content,content -> '$.key' AS 'key' FROM json_demo WHERE id = 3;

UPDATE json_demo 
SET content = JSON_SET(content, '$.temp2', 432)
WHERE id = 3;

SELECT id,content,content -> '$.key' AS 'key' FROM json_demo WHERE id = 3;

JSON陣列操作

查詢指定欄位值

SELECT
	id,
	content -> '$[*].key' AS 'key',
	content ->> '$[*].value' AS 'value',
	content -> '$[0].key' AS 'key2',
	content ->> '$[0].value' AS 'value2',
	/* 查詢陣列長度 */
	JSON_LENGTH(content) AS 'length'
FROM
	json_demo 
WHERE
	id < 3;

用於條件查詢

SELECT
	id,
	content -> '$[*].key' AS 'key',
	content ->> '$[*].value' AS 'value'
FROM
	json_demo 
WHERE
	id < 3
	/* content.value的值中存在like'%34%'的值 */
	AND content ->> '$[*].value' like '%34%'
	/* content.key的值中有4 */
	AND JSON_OVERLAPS(content ->> '$[*].key', '4' );

修改指定欄位值

基礎操作都跟JSON物件差不太多,就是在'$'後面加對應的索引位'$[0]',指定所有則'$[*]'。如果陣列中包含陣列,可以通過'$[1][2][3]'這種方式指定深層的陣列元素。

追加元素

JSON_ARRAY_APPEND和JSON_ARRAY_INSERT都可以實現陣列元素追加。區別在於JSON_ARRAY_APPEND可以不指定索引位,此時往最後位置追加;JSON_ARRAY_INSERT必須指定索引位,不指定則會報錯。

JSON_ARRAY_APPEND是追加在指定索引位後面,而JSON_ARRAY_INSERT則是插入到指定索引位前面。

更多操作

名稱描述
JSON_ARRAY()建立JSON陣列
JSON_ARRAY_APPEND()將資料附加到JSON檔案
JSON_ARRAY_INSERT()插入JSON陣列
JSON_CONTAINS()JSON檔案是否在路徑中包含特定物件
JSON_CONTAINS_PATH()JSON檔案是否在路徑中包含任何資料
JSON_DEPTH()JSON檔案的最大深度
JSON_EXTRACT()從JSON檔案返回資料
JSON_INSERT()將資料插入JSON檔案
JSON_KEYS()JSON檔案中的鍵陣列
JSON_LENGTH()JSON檔案中的元素數
JSON_MERGE() (已棄用)合併JSON檔案,保留重複的鍵。JSON_MERGE_PRESERVE()的已棄用同義詞
JSON_MERGE_PATCH()合併JSON檔案,替換重複鍵的值
JSON_MERGE_PRESERVE()合併JSON檔案,保留重複的鍵
JSON_OBJECT()建立JSON物件
JSON_OVERLAPS() (8.0.17引入)比較兩個JSON檔案,如果它們具有共同的任何鍵值對或陣列元素,則返回TRUE(1),否則返回FALSE(0)
JSON_PRETTY()以易於閱讀的格式列印JSON檔案
JSON_QUOTE()參照JSON檔案
JSON_REMOVE()從JSON檔案中刪除資料
JSON_REPLACE()替換JSON檔案中的值
JSON_SCHEMA_VALID() (8.0.17引入)根據JSON模式驗證JSON檔案;如果檔案針對架構進行了驗證,則返回TRUE / 1;否則,則返回FALSE / 0。
JSON_SCHEMA_VALIDATION_REPORT() (8.0.17引入)根據JSON模式驗證JSON檔案;以JSON格式返回有關驗證結果的報告,包括成功或失敗以及失敗原因
JSON_SEARCH()JSON檔案中值的路徑
JSON_SET()將資料插入JSON檔案
JSON_STORAGE_FREE()部分更新後,JSON列值的二進位制表示形式中的可用空間
JSON_STORAGE_SIZE()用於儲存JSON檔案的二進位制表示形式的空間
JSON_TABLE()從JSON表示式返回資料作為關係表
JSON_TYPE()JSON值型別
JSON_UNQUOTE()取消參照JSON值
JSON_VALID()JSON值是否有效
JSON_VALUE() (8.0.21引入)在提供的路徑所指向的位置從JSON檔案中提取值;以VARCHAR(512)或指定的型別返回此值
MEMBER OF() (8.0.17引入)如果第一個運算元與作為第二個運算元傳遞的JSON陣列的任何元素匹配,則返回true(1),否則返回false(0)

JSON欄位在JAVA中的實踐

這個是我的另外一篇關於JSON欄位實踐:

Mysql json型別欄位Java+Mybatis資料字典功能實踐

以上為個人經驗,希望能給大家一個參考,也希望大家多多支援it145.com。


IT145.com E-mail:sddin#qq.com