json - MySQL 5.7 - 选择 JSON 文档作为表
问题描述
我有以下 JSON 数据格式,其中包含简单的键/值对数据。
CREATE TABLE test_table (id INT, jdoc JSON);
INSERT INTO test_table VALUES
(1, '{"CS":15, "Physics":20,"Chemistry":10}'),
(2, '{"CS":6, "Physics":8,"Chemistry":5}');
我需要选择以下格式的上述数据。即键名(键名是动态的)作为列名,值作为数据。
id| CS | Physics | Chemistry
1 | 15 | 20 | 10
2 | 6 | 8 | 5
我从高级 SO 成员“Akina”那里得到了以下解决方案,使用JSON_TABLE
. 但是这个功能只支持 MySQL 8.0
https://stackoverflow.com/a/69673530/1791406
JSON_TABLE
MySQL 5.7 中是否有任何功能替代方案。或获得任何线索以获得所需的结果。
解决方案
是的,老式的方式,使用 json_extract 和路径:
https://dbfiddle.uk/?rdbms=mysql_5.7&fiddle=403c4a6fa81b514214d3f10ddb4cf50e
SELECT t.id,
t.jdoc->'$.CS' as CS,
t.jdoc->'$.Physics' as Physics,
t.jdoc->'$.Chemistry' as Chemistry
FROM test_table t
其中 -> 是 json_extract 的简写运算符。根据您获取的数据格式,您可能需要使用 ->>,它是 json_unquote(json_extract()) 的简写运算符。
您可以通过构建动态路径来支持未知键,但它的可读性会降低:
https://dbfiddle.uk/?rdbms=mysql_5.7&fiddle=bb9bf699ace41e6dd0e475e60b60605b
SELECT t.id,
json_extract(t.jdoc, concat('$.', json_extract(json_keys(t.jdoc), '$[0]'))) as k0,
json_extract(t.jdoc, concat('$.', json_extract(json_keys(t.jdoc), '$[1]'))) as k1,
json_extract(t.jdoc, concat('$.', json_extract(json_keys(t.jdoc), '$[2]'))) as k2
FROM test_table t;
您可以看到输出是相同的,除了列名。如果列名很重要,您可以使用准备好的语句动态生成带有字段名的查询,然后从中准备一条语句并执行它。看到这个小提琴:
https://dbfiddle.uk/?rdbms=mysql_5.7&fiddle=fb2ab27407789bdfad04e0a53d9a0257
SELECT json_keys(t.jdoc) INTO @keys FROM test_table t LIMIT 1;
SET @query = CONCAT("SELECT t.id, "
"t.jdoc->'$.", json_extract(@keys, '$[0]'), "' as ", json_extract(@keys, '$[0]'), ", ",
"t.jdoc->'$.", json_extract(@keys, '$[1]'), "' as ", json_extract(@keys, '$[1]'), ", ",
"t.jdoc->'$.", json_extract(@keys, '$[2]'), "' as ", json_extract(@keys, '$[2]'),
"FROM test_table t;");
PREPARE stmt FROM @query;
EXECUTE stmt;
可读性直接消失了,维护将是一件真正的苦差事。我相信变量已被弃用,所以不知道它会工作多长时间,通常建议避免运行动态生成的 sql。但它会产生预期的结果。
请记住,执行动态生成的 sql 会给您带来许多可能的危害。您实际上是从其他地方获取文本,然后在您的系统上执行它。
推荐阅读
- joi - 访问键包含点的 JOI ref
- openedge - 将一些表记录复制到同一张表中会产生一些问题
- sql-server - 在同一个表中导入平面文件而不是在新表中
- javascript - Blazor javascript 加载有时无法正常工作
- mysql - System.InvalidCastException:“从字符串“员工”到类型“布尔”的转换无效。
- c# - 如何在 ASP.NET 核心应用程序中使用 BackgroundService 每年执行一个方法?
- azure-functions - 无法在 MacBook Pro 上使用 HomeBrew 运行“monodevelop.azurefunctions.8.8”
- html - 使用 Flex box 和 ios 进行纵向横向切换
- hyperledger-fabric - Hyperledger Fabric 1.4 中不同对等点的区块链之间的差异
- python - 带有加法和乘法的python不理解代码