java - 一個JSONArray的根據里面的類型轉化為一個新的JSONArray
問題描述
現有一個LIST:
[{ 'type':'呼吸系統', 'illness':'肺氣腫', 'quotaName': '血壓'},{ 'type':'呼吸系統', 'illness':'肺氣腫', 'quotaName': '血常規'}, { 'type':'呼吸系統', 'illness':'哮喘', 'quotaName': '血常規'}, { 'type':'循環系統', 'illness':'高血壓', 'quotaName': '心電圖'}, { 'type':'循環系統', 'illness':'高血壓', 'quotaName': '心電彩超'} ]
我想得到的list:
[{ 'type':'呼吸系統', 'illnessList':[{ 'name':'肺氣腫', 'quotaList':[ { 'name':'血壓' }, { 'name':'血常規' } ]},{ 'name':'哮喘', 'quotaList':[{ 'name':'血常規'} ]} ]},{ 'type':'循環系統', 'illnessList':[{ 'name':'高血壓', 'quotaList':[{ 'name':'心電圖'},{ 'name':'心電彩超'} ]} ]} ]
原有的list,所有疾病系統和疾病,以及疾病檢測指標都是合在一起的
我想根據類型分類得到一個list,但始終找不到思路
循環,遍歷多次以后感覺腦袋糊掉了,請求各位大大給個思路
問題解答
回答1:按照題主的輸入輸出要求,從一個JSONArray轉換到另一個JSONArray...感覺輸出的JSONArray無非是按照了兩個屬性進行了歸類,其實結構應該類似于Map<String,Map<String,List<String>>,所以我的想法就是把輸入的JSONArray要轉化為Map<String, Map<String,List<String>>的結構即可...而看到我剛才說的按照屬性歸類...那...很顯然...噔噔噔!!!...心中自然浮現了Java8的Collectors.groupingBy...直接無腦groupingBy就行了嘛
下面是我的小思路和代碼:既然是面向對象,所以我先創建了一個輸入的Bo對象FromDataBo
@Getter@Setterpublic class FromDataBo { private String type; private String illness; private String quotaName;}
接著是按照輸出格式創建的輸出對象ToDataBo (帶注釋的方法先可以不看...只是轉化用的,可以先看數據結構)
@Getter@Setterpublic class ToDataBo { private String type; private List<ToDataIllnessBo> illnessList; /** * map轉化為List<ToDataBo> * @param map * @return */ public static List<ToDataBo> createByMap(Map<String, Map<String, List<String>>> map){return map.entrySet().stream().map(ToDataBo::of).collect(Collectors.toList()); } /** * 一個Map.Entry<String, Map<String, List<String>>>對應轉化為一個ToDataBo * @param entry * @return */ public static ToDataBo of(Map.Entry<String, Map<String, List<String>>> entry){ToDataBo dataBo = new ToDataBo();dataBo.setType(entry.getKey());dataBo.setIllnessList(entry.getValue().entrySet().stream().map(ToDataIllnessBo::of).collect(Collectors.toList()));return dataBo; }@Getter @Setter static class ToDataIllnessBo{private String name;private List<ToDataQuotaBo> quotaList;/** * 一個Map.Entry<String, List<String>>對應轉化為一個ToDataIllnessBo * @param entry * @return */public static ToDataIllnessBo of(Map.Entry<String, List<String>> entry){ ToDataIllnessBo dataIllnessBo = new ToDataIllnessBo(); dataIllnessBo.setName(entry.getKey()); dataIllnessBo.setQuotaList(entry.getValue().stream().map(ToDataQuotaBo::new).collect(Collectors.toList())); return dataIllnessBo;} } @Getter @Setter @AllArgsConstructor static class ToDataQuotaBo {private String name; }}
輸入輸出對象有了,那就可以進行最重要按屬性分類,我先把轉化為Map<String,Map<String,List<String>>的代碼貼出來...主要就是這個嘛...注釋嘛,熟悉lamdba的估計一眼就看出來...不熟悉再多了解哈吧
Map<String, Map<String, List<String>>> collect = fromDataBos.stream().collect(// 按照type分類Collectors.groupingBy(FromDataBo::getType,// 按照type分類后,同一類的數據再按照illness分類Collectors.groupingBy(FromDataBo::getIllness,// 按照type分類,再按照illness分類后,同一類的數據取其中的QuotaName并轉化為集合Collectors.mapping(FromDataBo::getQuotaName, Collectors.toList()))));
最后是完整的測試代碼和結果,最后的result對象就是你需要的輸出JSONArray
public class Test1 { public static void main(String[] args) {String from = '[n' +'{n' +' 'type':'呼吸系統',n' +' 'illness':'肺氣腫',n' +' 'quotaName': '血壓'n' +'},n' +'{n' +' 'type':'呼吸系統',n' +' 'illness':'肺氣腫',n' +' 'quotaName': '血常規'n' +'},n' +' {n' +' 'type':'呼吸系統',n' +' 'illness':'哮喘',n' +' 'quotaName': '血常規'n' +'},n' +' {n' +' 'type':'循環系統',n' +' 'illness':'高血壓',n' +' 'quotaName': '心電圖'n' +'},n' +' {n' +' 'type':'循環系統',n' +' 'illness':'高血壓',n' +' 'quotaName': '心電彩超'n' +'}n' +' ]';// 把輸入的JSONArray字符串轉化為FromDataBo集合List<FromDataBo> fromDataBos = JSONArray.parseArray(from, FromDataBo.class);// 歸類Map<String, Map<String, List<String>>> collect = fromDataBos.stream().collect(// 按照type分類Collectors.groupingBy(FromDataBo::getType,// 按照type分類后,同一類的數據再按照illness分類Collectors.groupingBy(FromDataBo::getIllness,// 按照type分類,再按照illness分類后,同一類的數據取其中的QuotaName并轉化為集合Collectors.mapping(FromDataBo::getQuotaName, Collectors.toList()))));// 歸類后的map轉化為輸出對象ToDataBo集合List<ToDataBo> toDataBos = ToDataBo.createByMap(collect);// 我是輸出對象,我在這JSONArray result = JSONArray.parseArray(JSONArray.toJSONString(toDataBos));System.out.println(result); }}
測試結果:
就醬...
回答2:這個不是后端吐出來的嗎?沒必要你去做處理啊,跟后端說下就好啊。或者你就用他們吐的,實現你的效果就好。無非你想更省事。直接一個循環處理。用上面的數組,在循環里做處理 跟你處理成后來的 是一樣的。不如前置處理。性能高。下面的是標準的風格。界面貼出來。
回答3:代碼地址http://jsbin.com/roqejoficu/e...
var convertData = function(data){ let result = []; var level1Obj = {}; var level2Obj = {}; var level3Obj = {}; data.forEach(function (item, index, arr) {//一層對象level1Obj[item.type] = {};level1Obj[item.type]['type'] = item.type;//2層對象level2Obj[item.type+item.illness] = {};level2Obj[item.type+item.illness]['p1'] = item.type;level2Obj[item.type+item.illness]['type'] = item.illness;//3層對象level3Obj[index] = {};level3Obj[index]['p1'] = item.type;level3Obj[index]['p2'] = item.illness;level3Obj[index]['type'] = item.quotaName; });for (var level1 in level1Obj) {var o1 = {};o1.type = level1Obj[level1].type;o1.list = [];result.push(o1);for (var level2 in level2Obj) { if (level2Obj[level2].p1 == level1Obj[level1].type) {var o2 = {};o2.type = level2Obj[level2].type;o2.list = [];o1.list.push(o2);for (var level3 in level3Obj) { if (level3Obj[level3].p1 == level1Obj[level1].type && level3Obj[level3].p2 == level2Obj[level2].type) {var o3 = {};o3.type = level3Obj[level3].type;o2.list.push(o3); }} }} } console.log(result); return result;},var result = convertData(data);
寫的不好,如有更優的方法,希望交流交流
相關文章:
1. javascript - git clone 下來的項目 想在本地運行 npm run install 報錯2. python - django 里自定義的 login 方法,如何使用 login_required()3. python如何不改動文件的情況下修改文件的 修改日期4. mysql主從 - 請教下mysql 主動-被動模式的雙主配置 和 主從配置在應用上有什么區別?5. angular.js - 不適用其他構建工具,怎么搭建angular1項目6. 主從備份 - 跪求mysql 高可用主從方案7. android-studio - Android 動態壁紙LayoutParams問題8. node.js - 使用 superagent 抓取 UTF-8網站亂碼9. mysql優化 - mysql count(id)查詢速度如何優化?10. sql語句如何按or排序取出記錄
