mysql5.6 解析JSON字符串方式(支持复杂的嵌套格式)
作者:邓1136507751 发布时间:2024-01-22 15:03:58
标签:mysql,JSON,字符串,嵌套
mysql5.6 解析JSON字符串
支持复杂的嵌套格式
废话不多说,先上代码。
CREATE FUNCTION `json_parse`(`jsondata` longtext,`keyname` text) RETURNS text CHARSET utf8
BEGIN
DECLARE delim VARCHAR(128);
DECLAREresult longtext;
DECLARE startpos INTEGER;
DECLARE endpos INTEGER;
DECLARE endpos1 INTEGER;
DECLARE findpos INTEGER;
DECLARE leftbrace INTEGER;
DECLARE tmp longtext;
DECLARE tmp2 longtext;
DECLARE Flag INTEGER;
SET delim = CONCAT('"', keyname, '": "');
SET startpos = locate(delim,jsondata);
IF startpos > 0 THEN
SET findpos = startpos+length(delim);
SET leftbrace = 1;
SET endpos = 0;
SET Flag =1;
get_token_loop: repeat
IF substr(jsondata,findpos,2)='\\"' THEN
SET findpos = findpos + 2;
iterate get_token_loop;
ELSEIF substr(jsondata,findpos,2)='\\\\' THEN
SET findpos = findpos + 2;
iterate get_token_loop;
ELSEIF substr(jsondata,findpos,1)='"' AND Flag = 1 THEN
SET endpos = findpos;
SET findpos = LENGTH(jsondata)+1;
leave get_token_loop;
END IF;
SET findpos = findpos + 1;
UNTIL findpos > LENGTH(jsondata) END repeat;
IF endpos > 0 THEN
SELECT
substr(
jsondata
,startpos
+length(delim)#取出value值的起始位置
,endpos#取出value值的结束位置
-(
startpos
+length(delim)
)#减去value值的起始位置,得到value值字符长度
) INTO result
FROM DUAL;
SET result= replace(result,'\\"','"');
SET result= replace(result,'\\\\','\\');
ELSE
SET result=null;
END IF;
/*
SELECT
substr(
jsondata
,locate(delim,jsondata)
+length(delim)#取出value值的起始位置
,locate(
'"'
,jsondata
,locate(delim,jsondata)
+length(delim)
)#取出value值的结束位置
-(
locate(delim,jsondata)
+length(delim)
)#减去value值的起始位置,得到value值字符长度
) INTO result
FROM DUAL;*/
ELSE
SET delim = CONCAT('"', keyname, '": {');
SET startpos = locate(delim,jsondata);
IF startpos > 0 THEN
SET findpos = startpos+length(delim);
SET leftbrace = 0;
SET endpos = 0;
SET Flag =0;
get_token_loop: repeat
IF substr(jsondata,findpos,2)='{"' THEN
SET leftbrace = leftbrace + 1;
SET findpos = findpos + 2;
iterate get_token_loop;
ELSEIF substr(jsondata,findpos,2)='\\"' THEN
SET findpos = findpos + 2;
iterate get_token_loop;
ELSEIF substr(jsondata,findpos,3)=': "' THEN
SET Flag = 1;
SET findpos = findpos + 3;
iterate get_token_loop;
ELSEIF substr(jsondata,findpos,1)='"' THEN
SET Flag = 0;
ELSEIF substr(jsondata,findpos,1)='}' AND Flag = 0 THEN
IF leftbrace > 0 THEN
SET leftbrace = leftbrace - 1;
ELSE
SET endpos = findpos;
SET findpos = LENGTH(jsondata)+1;
END IF;
END IF;
SET findpos = findpos + 1;
UNTIL findpos > LENGTH(jsondata) END repeat;
IF endpos > 0 THEN
SELECT
substr(
jsondata
,startpos
+length(delim)#取出value值的起始位置
,endpos#取出value值的结束位置
-(
startpos
+length(delim)
)#减去value值的起始位置,得到value值字符长度
) INTO result
FROM DUAL;
SET result=CONCAT("{",result, '}');
ELSE
SET result=null;
END IF;
ELSE
SET delim = CONCAT('"', keyname, '": [');
SET startpos = locate(delim,jsondata);
IF startpos > 0 THEN
SET findpos = startpos+length(delim);
SET leftbrace = 0;
SET endpos = 0;
SET tmp = substring_index(jsondata,delim,-1);
SET tmp2 = substring_index(tmp,']',1);
IF locate('[',tmp2) =0 THEN
SET endpos = locate(']',tmp);
SET endpos = endpos+findpos-1;
ELSE
get_token_loop: repeat
IF substr(jsondata,findpos,2)='\\"' THEN
SET findpos = findpos + 2;
iterate get_token_loop;
ELSEIF substr(jsondata,findpos,3)=': "' THEN
SET Flag = 1;
SET findpos = findpos + 3;
iterate get_token_loop;
ELSEIF substr(jsondata,findpos,1)='[' AND Flag = 0 THEN
SET leftbrace = leftbrace + 1;
SET findpos = findpos + 1;
iterate get_token_loop;
ELSEIF substr(jsondata,findpos,1)='"' THEN
SET Flag = 0;
ELSEIF substr(jsondata,findpos,1)=']' AND Flag = 0 THEN
IF leftbrace > 0 THEN
SET leftbrace = leftbrace - 1;
ELSE
SET endpos = findpos;
SET findpos = LENGTH(jsondata)+1;
END IF;
END IF;
SET findpos = findpos + 1;
UNTIL findpos > LENGTH(jsondata) END repeat;
END IF;
IF endpos > 0 THEN
SELECT
substr(
jsondata
,startpos
+length(delim)#取出value值的起始位置
,endpos#取出value值的结束位置
-(
locate(delim,jsondata)
+length(delim)
)#减去value值的起始位置,得到value值字符长度
) INTO result
FROM DUAL;
SET result=CONCAT("[",result, ']');
ELSE
SET result=null;
END IF;
ELSE
SET delim = CONCAT('"', keyname, '": ');
SET startpos = locate(delim,jsondata);
IF startpos > 0 THEN
SET endpos = locate(',',jsondata,startpos+length(delim));
SET endpos1 = locate('}',jsondata,startpos+length(delim));
IF endpos>0 OR endpos1>0 THEN
IF endpos1>0 AND endpos1 < endpos OR endpos =0 THEN
SET endpos = endpos1;
END IF;
SELECT
substr(
jsondata
,startpos
+length(delim)#取出value值的起始位置
,endpos#取出value值的结束位置
-(
locate(delim,jsondata)
+length(delim)
)#减去value值的起始位置,得到value值字符长度
) INTO result
FROM DUAL;
IF STRCMP(result,'null')=0 THEN
SET result=null;
END IF;
ELSE
SET result=null;
END IF;
ELSE
SET result=null;
END IF;
END IF;
END IF;
END IF;
if result='' and RIGHT(keyname,2)='Id' then
SET result=null;
end if;
RETURN result;
END
jsondata需要严格的json格式(注意逗号和分号以及双引号之间的空格)
SET jsondata='{"CurrentPage": 1, "data": [{"config": "123"}, {"config": "456"}], "PageSize": 10}'
SELECT json_parse(jsondata, 'CurrentPage') INTO CurrentPage;
SELECT json_parse(jsondata, 'data') INTO data;
这边如果想获取config的内容,可以这样处理
SET count = (LENGTH(data)-LENGTH(REPLACE(data,'},','')))/2+1;
SET i = 0;
WHILE i < count DO
SET SetObject = SUBSTRING_INDEX(SUBSTRING_INDEX(data,'},',i+1),'},',-1);
IF LENGTH(SetObject)>0 THEN
SELECT json_parse(SetObject, 'config') INTO config;
END IF;
SET i = i + 1;
END WHILE;
不足之处,jsondata数据多的情况下,会有效率问题。
mysql5.6及以下解析json方法
之前在公司发现在线的查询平台是MySQL5.6,不能用JSON_EXTRACT,也不能用存储过程,所以只能自己编了一个简单的小查询,几条数据还是能查的,如果数据量大的话,估计耗的资源就会比较多。
先说一下问题的背景
是想在'{"platform":"Android","source":"tt","details":null}'这一串东西里面找到source这个key对应的value值。
这个方法是先找到source":"这个字符串的起始位置和长度,这样就能够找到value值的起始位置;再找到这个字符串以后第一个"出现的位置,就能得到value值的结束位置。
再利用substr函数,就可以取出对应的位置。
下面是对应的代码
SELECT
'{"platform":"Android","source":"tt","details":null}' as 'sample'
,substr(
'{"platform":"Android","source":"tt","details":null}'
,locate('source":"','{"platform":"Android","source":"tt","details":null}')
+length('source":"')#取出value值的起始位置
,locate(
'"'
,'{"platform":"Android","source":"tt","details":null}'
,locate('source":"','{"platform":"Android","source":"tt","details":null}')
+length('source":"')
)#取出value值的结束位置
-(
locate('source":"','{"platform":"Android","source":"tt","details":null}')
+length('source":"')
)#减去value值的起始位置,得到value值字符长度
) as result
FROM DUAL
运行以后,就得到result的结果,就是tt。如果需要其他元素,就替换一下对应的key值和字段,就好了。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。
来源:https://blog.csdn.net/yt375230341/article/details/109643641


猜你喜欢
- 本文实例为大家分享了python实现简易聊天室的具体代码,供大家参考,具体内容如下群聊聊天室1.功能:类似qq群聊功能1.有人进入聊天室需要
- CREATE VIEW dbo.vw_db_dictionary AS SELECT TOP 100 PERCENT dbo.sysobje
- 目录前言super的用法super的原理Python super()使用注意事项混用super与显式类调用不同种类的参数总结前言Python
- 最近公司项目加了个页面,其中要求是这样的,点击对应列表,展开和收起,其实就是显示和隐藏内容部分;说来惭愧,我花了半天时间才搞出来(自黑一下~
- 前言有时候在使用Python处理比较耗时操作的时候,为了便于观察处理进度,这时候就需要通过进度条将处理情况进行可视化展示,以便我们能够及时了
- 一、构造dataframeimport pandas as pdimport numpy as npdf=pd.DataFrame(np.a
- 本文实例讲述了Python实现动态加载模块、类、函数的方法。分享给大家供大家参考,具体如下:动态加载模块: 方式1:系统函数__import
- 前言在学习过程中发现有时候交换字典的键和值,会使得我们最后的输出结果更加直观明了,整理出以下四种交换方式(data是原字典,new_data
- 桑基图桑基图(Sankey diagram),即桑基能量分流图,也叫桑基能量平衡图。它是一种特定类型的流程图,图中延伸的分支的宽度对应数据流
- 交互设计师主导或发起的项目很难拿到结果——现状这个问题在很多的小公司都不存在。小公司养着、催着设计师,设计师不用去考虑能不能拿到结果,因为你
- 【OpenCV】 ⚠️高手勿入! 半小时学会基本操作 ⚠️ 直线检测概述OpenCV 是一个跨平台的计算机视觉库, 支持多语言, 功能强大.
- golang并没有像C语言一样提供三元表达式。三元表达式的好处是可以用一行代码解决原本需要多行代码才能完成的功能,让冗长的代码瞬间变得简洁。
- 前几天读了下mysqld_safe脚本,个人感觉还是收获蛮大的,其中细致的交代了MySQL数据库的启动流程,包括查找MySQL相关目录,解析
- 写在前面今天在用爬虫及Pandas更新股票日线数据的时候发现KeyError报错,后面跟了一个DataFrame列索引,一开始以为是索引修改
- 目录背景实现工具打包调用总结众所周知,Golang 适合写 CLI 工具,但你可能还不知道 Golang 还可以打包配置文件。背景最近在写一
- 创建测试数据:import pandas as pdimport numpy as np#Create a DataFramedf1 = {
- 一、思路介绍在已有的单路径迷宫基础上打开一块合适的墙就可以构成2路径的迷宫。打开的墙不能和已有的路径过近。1。从开始和终点开始进行广度优先搜
- 在做自动化测试时,遇到两种情况需要判断usb是否已连接上(注,本文仅针对用adb命令来control手机)一种是在开测时(前提是同时要测试多
- 目录写在前面基本概念Windows搭建python开发环境从Hello World开始博客总结从大学开始玩python到现在参加工作,已经有
- 许多共享主机的服务提供商不允许运行你自己的服务进程,也不允许修改 httpd.conf 文件。 尽管如此,仍然有可能通过Web服务器产生的子