java無限遞歸遍歷json對象問題
需求
給一個json字符串,遍歷json字符串,輸出其每個key對應的value。
舉一個例子
{
"a":1,
"b":{
"b1":2
}
}能夠輸出:
a->1;b.b1->2
上例只是為了說明問題舉得一個最簡單的json,實際應用中json可以是嵌套非常復雜的結構。
方案
1、采用遞歸的方法打印每個路徑上的值
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對象
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對象
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對象中的每個值,但是丟失了json結構中key的路徑,此外對于數(shù)組接口,沒有輸出key。
對上面方法進行優(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對象
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中是一個json對象
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結構中key的每個路徑都保存了下來。
但是,還是有一點點的小瑕疵,對于數(shù)組結構,沒有輸出下標值。
2、json-flatter庫
json-flatter庫可以將json結構按照上面需求輸出,同時對于數(shù)組還能顯示下標。
例如:
{ "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}]總結
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
java線程并發(fā)blockingqueue類使用示例
BlockingQueue是一種特殊的Queue,若BlockingQueue是空的,從BlockingQueue取東西的操作將會被阻斷進入等待狀態(tài)直到BlocingkQueue進了新貨才會被喚醒,下面是用BlockingQueue來實現(xiàn)Producer和Consumer的例子2014-01-01
利用JAVA反射,讀取數(shù)據(jù)庫表名,自動生成對應實體類的操作
這篇文章主要介紹了利用JAVA反射,讀取數(shù)據(jù)庫表名,自動生成對應實體類的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-08-08
SpringBoot 任務調度動態(tài)設置方式(不用重啟服務)
這篇文章主要介紹了SpringBoot 任務調度 動態(tài)設置方式(不用重啟服務),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-11-11
Java 8 Stream.distinct() 列表去重的操作
這篇文章主要介紹了Java 8 Stream.distinct() 列表去重的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-12-12

