如何使用 Java 8 功能按值对 HashMap 进行升序和降序排序?

 

在上一篇文章中,我向您展示了如何通过键对Java 8中的Map进行排序 ,今天,我将教您如何使用Java 8功能(例如,lambda表达式,方法引用,流和新方法) 按值对Map进行排序。添加到java.util.Comparator和Map.Entry类中。 为了对任何Map进行排序,例如HashMap ,Hashtable,LinkedHashMap,TreemMap甚至ConcurrentHashMap ,您可以首先使用entrySet()方法获取条目集,然后可以通过调用stream()方法获取流。 entrySet()方法返回一个Set,该Set从java.util.Collection类继承stream()方法。 一旦获得流,就可以调用sorted()方法,该方法可以使用Comparator对Stream中可用的所有Map.Entry对象进行排序。

为了按值比较Map的条目,可以使用java.util.Map.Entry类中新添加的Map.Entry.comparingByValue()方法。

这与我们在上一篇文章中使用过的compareByKey()方法相对应。 这两种方法都被重载以与Comparable和Comparator对象一起使用。

对流进行排序后,您可以执行任何操作,例如,如果您只想按排序的顺序打印键,值或条目,则只需使用forEach()方法,或者如果要按值对Map进行排序,则可以您可以使用流类的collect()方法。

此方法接受收集器,并允许您将Stream的所有元素捕获到所需的任何集合中。 例如,如果要排序的地图,则可以使用java.util.stream.Collectors类的toMap()方法。

此方法是重载的,并提供了两种选择,例如,您可以收集任何类型的地图中的条目,也可以指定所需的地图类型,例如,为了使条目保持排序,我们将使用LinkedHashMap 。 它还允许您在相同值的情况下打破平局,例如,可以按想要的顺序排列它们。

顺便说一句,如果您好奇的话,您还可以查看Pluralsight的《 从Java 8使用Lambda表达式到从流到集合》课程,以了解有关特定于集合框架的Java 8新功能的更多信息。

如何使用 Java 8 功能按值对 HashMap 进行升序和降序排序?_第1张图片

简而言之,以下是在Java 8中按值升序或降序对HashMap进行排序的确切步骤,假设您已经有一个map对象

  1. 通过调用Map.entrySet()方法获取条目集
  2. 通过调用stream()方法获取条目流
  3. 用Comparator调用排序的方法
  4. 使用Map.Entry.comparingByValue()比较器按值对条目进行排序
  5. 使用collect()方法收集结果
  6. 使用Collectors.toMap()方法在另一个Map中获取结果。
  7. 在最后一个参数中提供LinkedHashMap :: new,以强制其返回LinkedHashMap,以保留排序顺序
  8. 为了按降序排序,只需使用Java 8的Collections.reverseOrder()或Comparator.reverse()方法来颠倒Comparator的顺序,有关添加到关键Java类中的新方法的完整列表,请参阅Java SE 8,对于Really Im耐心 。例如Java Collection Framework,String和Comparator等。

完成此步骤后,您将获得一个按值排序的Map。 现在您已经了解了理论和步骤,现在让我们看下一节中的代码示例以使其正确。

Java程序按值对地图排序

这是我们完整的Java程序,使用Java 8功能按值对Map进行排序,例如,通过使用接口上的默认方法和静态方法对Java 8中的现有类进行开发,从而对Java 8中现有类的新方法进行了排序。 在此示例中,我获得了项目地图及其费用(例如租金,公用事业,交通等)的Map。Map关键字是String ,代表项目,值是Integer,即费用。 我们的任务是按值对“地图”进行排序,以找出哪个项目花费最多,并按值的降序打印所有项目。

import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.Map;

import static java.util.stream.Collectors.*;
import static java.util.Map.Entry.*;

/*
 * Java Program to sort a Map by values in Java 8
 * 
 */
public class Main {

  public static void main(String[] args) throws Exception {

    // a Map with string keys and integer values
    Map budget = new HashMap<>();
    budget.put("clothes", 120);
    budget.put("grocery", 150);
    budget.put("transportation", 100);
    budget.put("utility", 130);
    budget.put("rent", 1150);
    budget.put("miscellneous", 90);

    System.out.println("map before sorting: " + budget);

    // let's sort this map by values first
    Map sorted = budget
        .entrySet()
        .stream()
        .sorted(comparingByValue())
        .collect(
            toMap(e -> e.getKey(), e -> e.getValue(), (e1, e2) -> e2,
                LinkedHashMap::new));

    System.out.println("map after sorting by values: " + sorted);

    // above code can be cleaned a bit by using method reference
    sorted = budget
        .entrySet()
        .stream()
        .sorted(comparingByValue())
        .collect(
            toMap(Map.Entry::getKey, Map.Entry::getValue, (e1, e2) -> e2,
                LinkedHashMap::new));

    // now let's sort the map in decreasing order of value
    sorted = budget
        .entrySet()
        .stream()
        .sorted(Collections.reverseOrder(Map.Entry.comparingByValue()))
        .collect(
            toMap(Map.Entry::getKey, Map.Entry::getValue, (e1, e2) -> e2,
                LinkedHashMap::new));

    System.out.println("map after sorting by values in descending order: "
        + sorted);
  }

}

Output
map before sorting: {grocery=150, utility=130, miscellneous=90, rent=1150,
 clothes=120, transportation=100}
map after sorting by values: {miscellneous=90, transportation=100,
 clothes=120, utility=130, grocery=150, rent=1150}
map after sorting by values in descending order: {rent=1150, grocery=150,
 utility=130, clothes=120, transportation=100, miscellneous=90}

 

您可以看到,在对地图进行排序之前,按照值将其随机排列,但是首先,我们对它们进行了按值的升序排序,然后对相同的Map进行了按值的降序排序,这就是为什么租金排在第一位的原因,它使我们付出了最高的代价。

一些技巧

1)使用静态导入可以缩短代码,可以静态导入Map.Entry和Collectors类。

2)尽可能使用方法引用代替lambda表达式。 请参阅本文章以了解更多有关如何lambda表达式转换为方法的引用在Java中8,如果你不熟悉的。

这就是如何在Java 8中按值对Map进行排序的全部内容。 您可以看到,使用添加到现有类的新方法对Map进行排序非常容易。 一切皆有可能,因为JDK 8的默认方法功能使您可以向现有类添加新方法。 在进行此增强之前,在Java中不破坏接口的现有客户端是不可能的,因为一旦将新方法添加到接口中,它的所有客户端都必须实现它。 如果方法是默认方法或静态方法,则不再需要此方法,因为它们不是抽象方法而是具体方法。

进一步阅读

Java 8的新增功能

Java SE 8实在不耐烦

使用Lambda表达式从Java 8中的集合到流

相关的Java 8教程

如果您有兴趣学习有关Java 8的新功能的更多信息,这是我以前的文章,介绍Java 8的一些重要概念:

  • 从零开始学习Java 8的5本书( 书籍 )
  • Java 8中的默认方法是什么? ( 示例 )
  • 如何在Java 8中连接String( 示例 )
  • 如何在Java 8中使用filter()方法( 教程 )
  • 如何在Java 8中使用LocalDateTime格式化/解析日期? ( 教程 )
  • 如何在Java 8中使用Stream类( 教程 )
  • 如何在Java 8中将列表转换为Map( 解决方案 )
  • Java 8中抽象类和接口之间的区别? ( 回答 )
  • Java 8中20个日期和时间的示例( 教程 )
  • 如何在Java 8中使用peek()方法( 示例 )
  • 如何在Java 8中按键对地图排序? ( 示例 )
  • 如何在Java 8中按值对may进行排序? ( 示例 )
  • Java 8中的Optionals的10个示例? ( 示例 )

感谢您到目前为止阅读本文。 如果您喜欢这篇文章,请与您的朋友和同事分享。 如果您有任何疑问或建议,请发表评论。

翻译自: https://www.javacodegeeks.com/2017/09/java-8-sorting-hashmap-values-ascending-descending-order.html

你可能感兴趣的:(如何使用 Java 8 功能按值对 HashMap 进行升序和降序排序?)