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包里的两个弄混了(因为是同名),这点大家还是要注意一哈:)

转载于:https://www.cnblogs.com/zhangyidadada/p/10572269.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值