将List中的元素按照属性分类成树状的map

Posted llawliet0001

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了将List中的元素按照属性分类成树状的map相关的知识,希望对你有一定的参考价值。

例如:一个学生信息的list,学生的信息由班级、性别、学号等,将list按照班级、性别分类。
只需:
     Map map = CollectionTools.classifyList(studentAllList, "classId","sex");
这个工具运用反射可将list中的每一个学生按照classId和sex分类成一个树状的map。

  1 import java.lang.reflect.InvocationTargetException;
  2 import java.util.ArrayList;
  3 import java.util.LinkedHashMap;
  4 import java.util.List;
  5 import java.util.Map;
  6 
  7 import org.apache.commons.beanutils.PropertyUtils;
  8 
  9 
 10 public class CollectionUtils {
 11 
 12     /**
 13      * 运用PropertyUtils取得bean的值,并根据keyName归类
 14      *
 15      * @param list    List beans
 16      * @param keyName 需要归类的bean的属性名称
 17      * @return LinkedHashMap<String , List>,有顺序的map<br>
 18      * map的key为需要归类的bean的属性名+"#"+对应的属性值:eg:"class#312"<br>
 19      * value为List<bean><br>
 20      * @throws IllegalAccessException
 21      * @throws InvocationTargetException
 22      * @throws NoSuchMethodException
 23      */
 24 
 25     public static LinkedHashMap<String, List> classify(List list, String keyName)
 26             throws IllegalAccessException, InvocationTargetException, NoSuchMethodException {
 27         LinkedHashMap<String, List> target = new LinkedHashMap();
 28         for (Object obj : list) {
 29             // 取得bean需要归类的属性(keyName)的值,不做类型转换
 30             Object oKeyValue = PropertyUtils.getProperty(obj, keyName);
 31             String keyValue = keyName + "#" + String.valueOf(oKeyValue);
 32             if (!target.containsKey(keyValue)) {
 33                 // 如果map中没有归类key值,则添加key值和相应的list
 34                 ArrayList keyList = new ArrayList();
 35                 keyList.add(obj);
 36                 target.put(keyValue, keyList);
 37             } else {
 38                 // 如果有归类key值,则在相应的list中添加这个bean
 39                 ArrayList keyList = (ArrayList) target.get(keyValue);
 40                 keyList.add(obj);
 41             }
 42         }
 43         return target;
 44     }
 45 
 46     /**
 47      * 将归类的Map<String, List>按照 keyName归类,并用index控制递归。<br>
 48      * 因为直接调用没有意义,这个方法为private,
 49      *
 50      * @param mocl     map of classified list<br>
 51      *                 也就是运用方法<br>
 52      *                 LinkedHashMap<String, List> classify(List list, String
 53      *                 keyName)<br>
 54      *                 将list归类成的map<br>
 55      * @param index    用条件 index < keyNames.length控制递归
 56      * @param keyNames 需要归类的bean的属性名称
 57      * @return
 58      * @throws IllegalAccessException
 59      * @throws InvocationTargetException
 60      * @throws NoSuchMethodException
 61      */
 62 
 63     private static LinkedHashMap<String, Map> classify(Map<String, List> mocl, int index,
 64                                                        String... keyNames) throws IllegalAccessException, InvocationTargetException,
 65             NoSuchMethodException {
 66         // 单步理解:target是函数参数Map<String, List> mocl再次归类成的LinkedHashMap<String,Map>
 67         // 递归到最后这个是最终归类好的map
 68         LinkedHashMap<String, Map> target = new LinkedHashMap();
 69         // 控制递归条件,起始的index应该总是1。
 70         if (index < keyNames.length) {
 71             // swap用来保存参数index的值,这是最容易出错的一个地方
 72             // 用它保证:在参数Map<String, List> mocl层面循环时用相同的index参数值。
 73             int swap = index;
 74             for (Map.Entry<String, List> entry : mocl.entrySet()) {
 75                 String mocl_key = entry.getKey();
 76                 List mocl_list = entry.getValue();
 77                 // 将List<bean>再次归类
 78                 LinkedHashMap<String, List> _mocl = classify(mocl_list, keyNames[index]);
 79                 // 如果index达到了数组的最后一个,一定是List<bean>转map,递归结束
 80                 if (index == keyNames.length - 1) {
 81                     target.put(mocl_key, _mocl);
 82                 } else {
 83                     // 将List<bean>转map得到的_mocl,再次归类
 84                     // _mocm 为map of classified map的简称
 85                     LinkedHashMap<String, Map> _mocm = classify(_mocl, ++index, keyNames);
 86                     target.put(mocl_key, _mocm);
 87                 }
 88                 index = swap;
 89             }
 90         }
 91         return target;
 92     }
 93 
 94     /**
 95      * 将Map<String, List> map按照bean需要归类的属性名keyName归类
 96      *
 97      * @param map     map of classified list<br>
 98      *                list归类成的map
 99      * @param keyName bean需要归类的属性名
100      * @return
101      * @throws IllegalAccessException
102      * @throws InvocationTargetException
103      * @throws NoSuchMethodException
104      */
105 
106     public static LinkedHashMap<String, Map> classifyMap(Map<String, List> map, String keyName)
107             throws IllegalAccessException, InvocationTargetException, NoSuchMethodException {
108         LinkedHashMap<String, Map> target = new LinkedHashMap();
109         for (Map.Entry<String, List> entry : map.entrySet()) {
110             List map_list = entry.getValue();
111             String map_key = entry.getKey();
112             LinkedHashMap<String, List> keyMap = classify(map_list, keyName);
113             target.put(map_key, keyMap);
114         }
115         return target;
116     }
117 
118     /**
119      * 将List<bean> 按照指定的bean的属性进行归类,keyNames的先后顺序会影响归类结果。<br>
120      * eg:一个学生列表,按照班级和性别归类<br>
121      * Map map = CollectionUtils.classifyList(studentList, "classId","sex");<br>
122      *
123      * @param list     List beans
124      * @param keyNames 数组包含需要归类的bean的属性名称
125      * @return 归类的有顺序的树状map<br>
126      * map的key为需要归类的bean的属性名+"#"+对应的属性值:eg:"class#312"<br>
127      * map的值为List或者map
128      * @throws IllegalAccessException
129      * @throws InvocationTargetException
130      * @throws NoSuchMethodException
131      */
132     public static LinkedHashMap classifyList(List list, String... keyNames)
133             throws IllegalAccessException, InvocationTargetException, NoSuchMethodException {
134         if (keyNames == null || keyNames.length == 0)
135             return null;
136         if (keyNames.length == 1)
137             return classify(list, keyNames[0]);
138         else
139             return classify(classify(list, keyNames[0]), 1, keyNames);
140     }
141 }

 




以上是关于将List中的元素按照属性分类成树状的map的主要内容,如果未能解决你的问题,请参考以下文章

java arraylist 按照对象的属性排序

Android中对list的日期元素进行排序

四Collectors

java中list和map的区别

php中使用array_slice将数组中的元素分类

java中如何map转换为数组