java無限遞歸遍歷json對(duì)象問題
需求
給一個(gè)json字符串,遍歷json字符串,輸出其每個(gè)key對(duì)應(yīng)的value。
舉一個(gè)例子
{ "a":1, "b":{ "b1":2 } }
能夠輸出:
a->1;b.b1->2
上例只是為了說明問題舉得一個(gè)最簡(jiǎn)單的json,實(shí)際應(yīng)用中json可以是嵌套非常復(fù)雜的結(jié)構(gòu)。
方案
1、采用遞歸的方法打印每個(gè)路徑上的值
static final String json_schema2 = "{\"a\":1,\"b\":{\"b1\":2},\"c\":[1,2,3,4],\"d\":{\"d1\":{\"d1_1\":100,\"d1_2\":101},\"d2\":[\"d2_1\",\"d2_2\"]},\"e\":[{\"e1\":1},{\"e2\":2}]}"; static void analysisJson(Object objJson){ //如果objJson為json數(shù)組 if(objJson instanceof JSONArray) { JSONArray objArray = (JSONArray)objJson; for (int i = 0; i < objArray.size(); i++) { analysisJson(objArray.get(i)); } } else if(objJson instanceof JSONObject) { //如果objJson為json對(duì)象 JSONObject jsonObject = (JSONObject)objJson; Iterator it = jsonObject.keySet().iterator(); while(it.hasNext()) { String key = it.next().toString(); Object value = jsonObject.get(key); //value if(value instanceof JSONArray) { //如果value是數(shù)組 JSONArray objArray = (JSONArray)value; analysisJson(objArray); } else if(value instanceof JSONObject){ //如果value是json對(duì)象 analysisJson((JSONObject)value); } else { //如果value是基本類型 System.out.println("["+key+"]:"+value.toString()+" "); } } } else { //objJson為基本類型 System.out.println(objJson.toString()+" "); } }
輸出:
[a]:1
[b1]:2
1
2
3
4
[d1_1]:100
[d1_2]:101
d2_1
d2_2
[e1]:1
[e2]:2
上面的方法雖然可以輸出json對(duì)象中的每個(gè)值,但是丟失了json結(jié)構(gòu)中key的路徑,此外對(duì)于數(shù)組接口,沒有輸出key。
對(duì)上面方法進(jìn)行優(yōu)化:
static void analysisJson2(Object objJson,String flag) { if(objJson instanceof JSONArray) {//如果obj為json數(shù)組 JSONArray objArray = (JSONArray)objJson; for (int i = 0; i < objArray.size(); i++) { analysisJson2(objArray.get(i),flag); } } else if(objJson instanceof JSONObject){//如果為json對(duì)象 JSONObject jsonObject = (JSONObject)objJson; Iterator it = jsonObject.keySet().iterator(); while(it.hasNext()) { String key = it.next().toString(); Object object = jsonObject.get(key); //如果得到的是數(shù)組 if(object instanceof JSONArray){ JSONArray objArray = (JSONArray)object; String path = ""; if (StringUtils.isNotBlank(flag)) { path = flag + "." + key; } else { path = key; } analysisJson2(objArray,path); } else if(object instanceof JSONObject) {//如果key中是一個(gè)json對(duì)象 String path = ""; if (StringUtils.isNotBlank(flag)) { path = flag + "." + key; } else { path = key; } analysisJson2((JSONObject)object,path); } else {//如果key中是其他 String path = ""; if (StringUtils.isNotBlank(flag)) { path = flag + "." + key; } else { path = key; } System.out.println(path+":"+object.toString()+" "); } } } else {//如果key中是其他 System.out.println(flag+":"+objJson.toString()+" "); } }
輸出:
a:1
b.b1:2
c:1
c:2
c:3
c:4
d.d1.d1_1:100
d.d1.d1_2:101
d.d2:d2_1
d.d2:d2_2
e.e1:1
e.e2:2
非常完美,吧json結(jié)構(gòu)中key的每個(gè)路徑都保存了下來。
但是,還是有一點(diǎn)點(diǎn)的小瑕疵,對(duì)于數(shù)組結(jié)構(gòu),沒有輸出下標(biāo)值。
2、json-flatter庫(kù)
json-flatter庫(kù)可以將json結(jié)構(gòu)按照上面需求輸出,同時(shí)對(duì)于數(shù)組還能顯示下標(biāo)。
例如:
{ "a": { "b": 1, "c": null, "d": [false, true] }, "e": "f", "g": 2.3 } //輸出 { "a.b": 1, "a.c": null, "a.d[0]": false, "a.d[1]": true, "e": "f", "g": 2.3 }
示例:
<dependency> <groupId>com.github.wnameless.json</groupId> <artifactId>json-flattener</artifactId> <version>0.9.0</version> </dependency>
String json = "{ \"a\" : { \"b\" : 1, \"c\": null, \"d\": [false, true] }, \"e\": \"f\", \"g\":2.3 }"; Map<String, Object> flattenJson = JsonFlattener.flattenAsMap(json); System.out.println(flattenJson); // Output: {a.b=1, a.c=null, a.d[0]=false, a.d[1]=true, e=f, g=2.3} String jsonStr = JsonFlattener.flatten(json); System.out.println(jsonStr); // Output: {"a.b":1,"a.c":null,"a.d[0]":false,"a.d[1]":true,"e":"f","g":2.3} String nestedJson = JsonUnflattener.unflatten(jsonStr); System.out.println(nestedJson); // {"a":{"b":1,"c":null,"d":[false,true]},"e":"f","g":2.3} // Support JSON keys which contain dots or square brackets String flattendJsonWithDotKey = JsonFlattener.flatten("[{\"a.a.[\":1},2,{\"c\":[3,4]}]"); System.out.println(flattendJsonWithDotKey); // Output: {"[0][\"a.a.[\"]":12,"[1]":2,"[2].c[0]":3,"[2].c[1]":4} String nestedJsonWithDotKey = JsonUnflattener.unflatten( "{\"[1][0];\":2,\"[0]\":1,\"[1][1]\":3,\"[2]\":4,\"[3][\\\"ab.c.[\\\"]\":5}"); System.out.println(nestedJsonWithDotKey); // Output: [1,[2,3],4,{"ab.c.[":5}]
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
java線程并發(fā)blockingqueue類使用示例
BlockingQueue是一種特殊的Queue,若BlockingQueue是空的,從BlockingQueue取東西的操作將會(huì)被阻斷進(jìn)入等待狀態(tài)直到BlocingkQueue進(jìn)了新貨才會(huì)被喚醒,下面是用BlockingQueue來實(shí)現(xiàn)Producer和Consumer的例子2014-01-01利用JAVA反射,讀取數(shù)據(jù)庫(kù)表名,自動(dòng)生成對(duì)應(yīng)實(shí)體類的操作
這篇文章主要介紹了利用JAVA反射,讀取數(shù)據(jù)庫(kù)表名,自動(dòng)生成對(duì)應(yīng)實(shí)體類的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-08-08SpringBoot 任務(wù)調(diào)度動(dòng)態(tài)設(shè)置方式(不用重啟服務(wù))
這篇文章主要介紹了SpringBoot 任務(wù)調(diào)度 動(dòng)態(tài)設(shè)置方式(不用重啟服務(wù)),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11SpringMvc使用GoogleKaptcha生成驗(yàn)證碼
這篇文章主要為大家詳細(xì)介紹了SpringMvc項(xiàng)目中使用GoogleKaptcha 生成驗(yàn)證碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-09-09Java實(shí)現(xiàn)貪吃蛇游戲(1小時(shí)學(xué)會(huì))
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)貪吃蛇游戲,1小時(shí)學(xué)會(huì)貪吃蛇游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-05-05Java 8 Stream.distinct() 列表去重的操作
這篇文章主要介紹了Java 8 Stream.distinct() 列表去重的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-12-12IDEA自動(dòng)生成類圖和時(shí)序圖的操作指南
idea 的強(qiáng)大之處在于此,它包含了很多小插件,我們不需要再次下載相關(guān)插件,只需要在idea中小小的設(shè)置一下就可以了,本文我介紹了IDEA自動(dòng)生成類圖和時(shí)序圖的操作指南,我用的是idea2020版本,需要的朋友可以參考下2024-05-05