我想使用比较器按降序对对象进行排序。
class Person {
private int age;
}
这里我想对Person对象数组进行排序。
我该怎么做?
答案 0 :(得分:99)
您可以通过这种方式执行降序排序的用户定义类,覆盖compare()方法,
Collections.sort(unsortedList,new Comparator<Person>() {
@Override
public int compare(Person a, Person b) {
return b.getName().compareTo(a.getName());
}
});
或使用Collection.reverse()
按his comment中提到的用户 Prince 降序排序。
你可以像这样进行升序排序,
Collections.sort(unsortedList,new Comparator<Person>() {
@Override
public int compare(Person a, Person b) {
return a.getName().compareTo(b.getName());
}
});
用Lambda表达式替换上面的代码(Java 8以后)我们简明扼要:
Collections.sort(personList, (Person a, Person b) -> b.getName().compareTo(a.getName()));
从Java 8开始,List有sort()方法,以Comparator为参数(更简洁):
personList.sort((a,b)->b.getName().compareTo(a.getName()));
此处a
和b
通过lambda表达式推断为Person类型。
答案 1 :(得分:63)
对于什么,它的价值在于我的标准答案。这里唯一新的是使用Collections.reverseOrder()。此外,它将所有建议放在一个例子中:
/*
** Use the Collections API to sort a List for you.
**
** When your class has a "natural" sort order you can implement
** the Comparable interface.
**
** You can use an alternate sort order when you implement
** a Comparator for your class.
*/
import java.util.*;
public class Person implements Comparable<Person>
{
String name;
int age;
public Person(String name, int age)
{
this.name = name;
this.age = age;
}
public String getName()
{
return name;
}
public int getAge()
{
return age;
}
public String toString()
{
return name + " : " + age;
}
/*
** Implement the natural order for this class
*/
public int compareTo(Person p)
{
return getName().compareTo(p.getName());
}
static class AgeComparator implements Comparator<Person>
{
public int compare(Person p1, Person p2)
{
int age1 = p1.getAge();
int age2 = p2.getAge();
if (age1 == age2)
return 0;
else if (age1 > age2)
return 1;
else
return -1;
}
}
public static void main(String[] args)
{
List<Person> people = new ArrayList<Person>();
people.add( new Person("Homer", 38) );
people.add( new Person("Marge", 35) );
people.add( new Person("Bart", 15) );
people.add( new Person("Lisa", 13) );
// Sort by natural order
Collections.sort(people);
System.out.println("Sort by Natural order");
System.out.println("\t" + people);
// Sort by reverse natural order
Collections.sort(people, Collections.reverseOrder());
System.out.println("Sort by reverse natural order");
System.out.println("\t" + people);
// Use a Comparator to sort by age
Collections.sort(people, new Person.AgeComparator());
System.out.println("Sort using Age Comparator");
System.out.println("\t" + people);
// Use a Comparator to sort by descending age
Collections.sort(people,
Collections.reverseOrder(new Person.AgeComparator()));
System.out.println("Sort using Reverse Age Comparator");
System.out.println("\t" + people);
}
}
答案 2 :(得分:16)
我会为person类创建一个比较器,可以使用某种排序行为进行参数化。在这里,我可以设置排序顺序,但也可以修改它以允许对其他人属性进行排序。
public class PersonComparator implements Comparator<Person> {
public enum SortOrder {ASCENDING, DESCENDING}
private SortOrder sortOrder;
public PersonComparator(SortOrder sortOrder) {
this.sortOrder = sortOrder;
}
@Override
public int compare(Person person1, Person person2) {
Integer age1 = person1.getAge();
Integer age2 = person2.getAge();
int compare = Math.signum(age1.compareTo(age2));
if (sortOrder == ASCENDING) {
return compare;
} else {
return compare * (-1);
}
}
}
(希望它现在编译,我手头没有IDE或JDK,编码'盲目')
修改强>
感谢Thomas,编辑了代码。我不会说Math.signum的用法是好的,高性能的,有效的,但是我想保留它作为提示,compareTo方法可以返回任何整数,乘以(-1)将失败,如果实现返回Integer.MIN_INTEGER ...我删除了setter,因为它足够便宜,可以在需要时构建一个新的PersonComparator。
但我保持拳击,因为它表明我依赖现有的可比较实现。本可以做Comparable<Integer> age1 = new Integer(person1.getAge());
之类的事,但看起来太难看了。这个想法是展示一个可以很容易地适应其他人物属性的模式,比如名字,生日作为日期等等。
答案 3 :(得分:14)
String[] s = {"a", "x", "y"};
Arrays.sort(s, new Comparator<String>() {
@Override
public int compare(String o1, String o2) {
return o2.compareTo(o1);
}
});
System.out.println(Arrays.toString(s));
-> [y, x, a]
现在你必须为你的Person类实现Comparator。
类似于(按升序排列):compare(Person a, Person b) = a.id < b.id ? -1 : (a.id == b.id) ? 0 : 1
或Integer.valueOf(a.id).compareTo(Integer.valueOf(b.id))
。
为了最大限度地减少混淆,您应该实施升序比较器并使用包装器(like this)new ReverseComparator<Person>(new PersonComparator())
将其转换为降序比较器。
答案 4 :(得分:5)
使用Google Collections:
class Person {
private int age;
public static Function<Person, Integer> GET_AGE =
new Function<Person, Integer> {
public Integer apply(Person p) { return p.age; }
};
}
public static void main(String[] args) {
ArrayList<Person> people;
// Populate the list...
Collections.sort(people, Ordering.natural().onResultOf(Person.GET_AGE).reverse());
}
答案 5 :(得分:0)
java.util.Collection的类有一个排序方法,它采用列表和自定义Comparator。您可以定义自己的Comparator来按您喜欢的方式对Person对象进行排序。
答案 6 :(得分:0)
package com.test;
import java.util.Arrays;
public class Person implements Comparable {
private int age;
private Person(int age) {
super();
this.age = age;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public int compareTo(Object o) {
Person other = (Person)o;
if (this == other)
return 0;
if (this.age < other.age) return 1;
else if (this.age == other.age) return 0;
else return -1;
}
public static void main(String[] args) {
Person[] arr = new Person[4];
arr[0] = new Person(50);
arr[1] = new Person(20);
arr[2] = new Person(10);
arr[3] = new Person(90);
Arrays.sort(arr);
for (int i=0; i < arr.length; i++ ) {
System.out.println(arr[i].age);
}
}
}
这是一种方法。