Introduction
Sorting data is a common task in many software development projects. When working with collections of objects in Java, a powerful and flexible approach to sorting is to use the Comparator.comparing interface in conjunction with Streams. In this post, we are going to show that using Comparator.comparing to sort Java Stream can make sorting elegant and efficient.
What is the Comparator.comparing interface?
The Comparator.comparing interface is a feature introduced in Java 8 as part of the java.util.Comparator. package. It provides a static method called comparing that allows you to specify a key function (sort key) to compare objects. This function is used to extract a value from an object and compare it against that value during sorting.
Flexibility in sorting with Comparator.comparing
One of the main advantages of the Comparator.comparing interface is its flexibility. With it, we can perform sorting in different fields of an object, allowing the creation of complex sorting logic in a simple and concise way. Notice in the code below that simply in the sorted() method, we pass the Comparator.comparing interface as an argument, which in turn, passes the city field as an argument using method reference (People::getCity) performing the sort by this field.
class People {
public String name;
public String city;
public String getName(){
return name;
}
public String getCity(){
return city;
}
public People(String name, String city){
this.name = name;
this.city = city;
}
}
List.of(new People("John", "Boston"),
new People("Mary", "Boston"),
new People("Anthony", "Boston"),
new People("Monica", "Amsterdam"),
new People("Seth", "São Paulo"))
.stream()
.sorted(Comparator.comparing(People::getCity))
.map(People::getName)
.forEach(System.out::println);
}
Output
Monica
John
Mary
Anthony
Seth
Multi-criteria ordering
Often, it is necessary to perform sorting based on multiple criteria. This is easily achieved with the Comparator.comparing. interface by simply chaining together several comparing methods, each specifying a different criterion. Java will carry out the ordering according to the specified sequence. For example, we can sort the same list by city and then by name: Comparator.comparing(People::getCity).thenComparing(People:: getName).
List.of(new People("John", "Boston"),
new People("Mary", "Boston"),
new People("Anthony", "Boston"),
new People("Monica", "Amsterdam"),
new People("Seth", "São Paulo"))
.stream()
.sorted(Comparator.comparing(People::getCity)
.thenComparing(People::getName))
.map(People::getName)
.forEach(System.out::println);
Ascending and descending sort
Another important advantage of the Comparator.comparing interface is the ability to perform sorting in both ascending and descending order. To do this, just chain the method reversed() as in the code below:
List.of(new People("John", "Boston"),
new People("Mary", "Boston"),
new People("Anthony", "Boston"),
new People("Monica", "Amsterdam"),
new People("Seth", "São Paulo"))
.stream()
.sorted(Comparator.comparing(People::getCity).reversed()
.thenComparing(Comparator.comparing(People::getName).reversed()))
.map(People::getName)
.forEach(System.out::println);
Output
Seth
Mary
John
Anthony
Monica
Efficiency and simplicity
By using the Comparator.comparing interface in conjunction with Streams, sorting becomes more efficient and elegant. The combination of these features allows you to write clean code that is easy to read and maintain. Furthermore, Java internally optimizes sorting using efficient algorithms, resulting in satisfactory performance even for large datasets.
Final conclusion
The Comparator.comparing interface is a powerful tool to perform the sorting of Streams in Java. Its flexibility, ascending and descending sorting capabilities, support for multiple criteria, and efficient execution make it a valuable choice for any Java developer. By taking advantage of this interface, we can obtain a more concise, less verbose and efficient code, facilitating the manipulation of objects in a Stream.
Hope you enjoyed!