java8 集合处理

List to Map

   list 转 map 这个还是挺常用的,在java8下可以用收集器很容易就做到,java7的话使用googleguava好像也挺不错,不过我实在不喜欢匿名内部类的写法,所以索性就自己写个啦。so let's start

        //小明和小红的一些考试成绩,该类的属性分别是[id,学生姓名,考试科目,考试成绩]
        List<StudentScore> studentList = Arrays.asList(
                new StudentScore(1,"小明","语文",85),
                new StudentScore(2,"小明","数学",90),
                new StudentScore(3,"小明","英语",80),
                new StudentScore(4,"小红","语文",80),
                new StudentScore(5,"小红","数学",80),
                new StudentScore(6,"小红","英语",80));
        //现在要将集合转换成为 学生姓名 ->[科目] 的map集合
        //例如[小明]->[语文,数学,英语] 这样的map集合,该怎么办咧
        

Java8 Ways

  这种情况在做一些集合处理数据的时候有时会出现,那么首先是java8下的写法

        //转换(以下代码静态导入了java.util.stream.Collectors下的所有静态方法)
        Map<String, List<String>> stuMap = studentScoreList.stream()
        .collect(groupingBy(StudentScore::getStudentName
        , mapping(StudentScore::getSubjectName, toList())));
        //输出
        stuMap.forEach((s, strings) -> System.out.println("key: " + s +"\t value: " + strings));
        
输出结果:
key: 小明  value: [语文, 数学, 英语]
key: 小红  value: [语文, 数学, 英语]

   java8在前面的博客已经有介绍了,这里简单说一下,代码第三行将集合按照学生姓名进行了分组,第4行使用了下游收集器,将学生的考试科目名称进行了收集,并且是以list的集合形式收集的,因此就做到了以上的输出效果。
不得不承认,java8的流操作包办了几乎一切集合的操作,确实方便,那么在java7中该怎么做呢,我自己写了工具方法,java7环境要转换的话直接调用就好啦。

Java7 Ways

        //转换,三个参数分别是[要转换的集合,作为key值的属性名,作为value值的属性名]
        Map<String, List<String>> stuMapJava7 = CollectionUtils.listToMap(studentScoreList, "studentName", "subjectName");
        for (Map.Entry<String, List<String>> entry : stuMapJava7.entrySet()) {
            System.out.println("key: " + entry.getKey() +"\t value: " + entry.getValue());
        }
输出结果:
key: 小明  value: [语文, 数学, 英语]
key: 小红  value: [语文, 数学, 英语]

  这里我提供了两个重载方法,一个就是上面演示的三个参数的分别是[要转换的集合,作为key值的属性名,作为value值的属性名],另一个方法提供两个参数分别是[要转的集合,作为key值的属性名],另外一个方法的value值就是对象本身了。下面是代码,方法中我认为比较巧妙的一点是通过对list集合地址内容的修改来完成相关集合的生成。

  /**
       * 该方法用于list转map的重载方法,可自定义map映射的属性值 by LDF
       * @param list            用于转换的初始集合list
       * @param key             用于分组的key值,key值可以不唯一,不唯一的话类似于数据库的groupBy操作进行分组
       * @param valueProperName value值的属性名
       * @param <T>             初始集合list中的对象的泛型
       * @param <K>             转换后map集合的value值的泛型
       * @return 形如 key -> [valueProperName] 的map集合
       */
      public static <T, K> Map<String, List<K>> listToMap(List<T> list, String key, String valueProperName) {
            Map<String, List<K>> returnMap = new HashMap<>();
            try {
                  for (T t : list) {
                        Field name = t.getClass().getDeclaredField(key);//通过反射获得私有属性,这里捕获获取不到属性异常
                        name.setAccessible(true);//获得访问和修改私有属性的权限
                        String keyName = name.get(t).toString();//获得key值
                        List<K> tempList = returnMap.get(keyName);
                        if (tempList == null) {
                              tempList = new ArrayList<>();
                              Field field = t.getClass().getDeclaredField(valueProperName);//同上,通过反射拿到私有属性
                              field.setAccessible(true);
                              K k = (K) field.get(t);//强转,这里抛出转换异常
                              tempList.add(k);//这里的添加已经同步影响到map集合了,因为引用的是地址
                              returnMap.put(keyName, tempList);
                        } else {
                              Field field = t.getClass().getDeclaredField(valueProperName);
                              field.setAccessible(true);
                              K k = (K) field.get(t);
                              tempList.add(k);//这里的添加已经同步影响到map集合了,因为引用的是地址
                        }
                  }
            } catch (NoSuchFieldException | IllegalAccessException e) {
                  e.printStackTrace();
            }
            return returnMap;
      }
      
        /**
     * <h1 style="color:#007979"> 根据多属性进行list转map分组</h1>
     *
     * @param list    要转换的集合 by LDF
     * @param strings 作为key的属性名,这里可以指定多个属性哦,用逗号分开就可以了,例如要指定名字和年龄都相同的为一组(假设你要转换的集合叫list)参数这里就 填写(list, "name", "age")
     * @param <T>     集合里对象的泛型
     * @return
     */
    public static <T> Map<String, List<T>> listToMap(List<T> list, String... strings) {
        Map<String, List<T>> returnMap = new HashMap<>();
        try {
            for (T t : list) {
                StringBuffer stringBuffer = new StringBuffer();
                for (String s : strings) {
                    Field name1 = t.getClass().getDeclaredField(s);//通过反射获得私有属性,这里捕获获取不到属性异常
                    name1.setAccessible(true);//获得访问和修改私有属性的权限
                    String key = name1.get(t).toString();//获得key值
                    stringBuffer.append(key);
                }
                String KeyName = stringBuffer.toString();

                List<T> tempList = returnMap.get(KeyName);
                if (tempList == null) {
                    tempList = new ArrayList<>();
                    tempList.add(t);
                    returnMap.put(KeyName, tempList);
                } else {
                    tempList.add(t);
                }
            }
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
        return returnMap;
    }

filter 过滤

  • java8Ways
        List<StudentScore> filterStuListJava8 = studentScoreList
                .stream()
                .filter(s -> s.getScore()>=85)
                .collect(toList());
        System.out.println(filterStuListJava8);

输出结果

[StudentScore{id=1, studentName='小明', subjectName='语文', score=85},
 StudentScore{id=2, studentName='小明', subjectName='数学', score=90}]

声明式的链式代码看的确实很爽,那么在java7该如何实现呢

  • java7Ways
    filter两个参数分别是[需要过滤的集合,过滤条件],返回[过滤后的集合]
        List<StudentScore> filterStuListMy = filter(studentScoreList, new IPredicate<StudentScore>() {
            @Override
            public boolean apply(StudentScore studentScore) {
                return studentScore.getScore() >= 85;
            }
        });
        System.out.println(filterStuListMy);

输出结果

[StudentScore{id=1, studentName='小明', subjectName='语文', score=85},
 StudentScore{id=2, studentName='小明', subjectName='数学', score=90}]

那么是如何实现这样的效果的咧
依旧是写了辅助类来实现声明式的效果
首先是一个函数接口

//@FunctionalInterface(java7 中没有该注释)
public interface IPredicate<T> {
    boolean apply(T t);
}

然后是一个实现方法

   /**
     * 该方法接受一个需要过滤的集合和一个过滤条件(通过重写接口的apply方法来定义条件) by LDF
     * @param t 需要过滤的集合
     * @param iPredicate 过滤的条件
     * @param <T> 集合的泛型类
     * @return 过滤后的集合
     */
    public static <T> List<T> filter(List<T> t, IPredicate<? super T> iPredicate){

        List<T> returnList = new ArrayList<>();
        for (T t1 : t) {
            if (iPredicate.apply(t1)) {
                returnList.add(t1);
            }
        }
        return returnList;
    }

ps:你还可以使用google的guava集合类库完成过滤操作,操作方法十分相似,区别是guava的返回值为Collection,而自定义的方法就比较灵活了

带条件的distinc

java8流操作的已经包含了distinc了,但是该distinc是不带条件的,如果想要根据集合的某一个属性来去重,该怎么办咧?代码如下
例如要根据学生姓名来去重

        //去重
        studentScoreList.stream()
                .filter(distinctByKey(StudentScore::getStudentName))
                .forEach(System.out::println);

去重方法如下,利用一个map集合将第一次看见的值放入,并标记为true,然后用该map里的值与Null作对比(因为如果是null的话就证明这条数据没出现过,因此是非重复的数据,可以通过过滤)

备注:该方法来自于Oracle的Stuart Marks(java8的开发者)与 Tagir Valeev(IntelliJ IDEA的开发者)的修正

    /**
     * 该方法根据集合中对象的某一个属性进行去重
     * @param keyExtractor 去重的条件属性
     * @param <T>
     * @return
     */
    public static <T> Predicate<T> distinctByKey(Function<? super T, ?> keyExtractor) {
        Map<Object,Boolean> seen = new ConcurrentHashMap<>();
        return t -> seen.putIfAbsent(keyExtractor.apply(t), Boolean.TRUE) == null;}

tip:写例子的时候有时候发现function或者Predicate的接口参数编译器总是不通过,找了半天发现原来是guava包里的两个弄混了(因为是同名),这点大家还是要注意一哈:)

posted @ 2019-03-21 15:58  张益达达达  阅读(223)  评论(0编辑  收藏  举报