Подтвердить что ты не робот

Сортировка с использованием Comparator-Descending order (Пользовательские классы)

Я хочу сортировать объекты в порядке убывания с помощью компаратора.

class Person {
 private int age;
}

Здесь я хочу отсортировать массив объектов Person.

Как я могу это сделать?

4b9b3361

Ответ 1

Вы можете сделать нисходящий вид определяемого пользователем класса таким образом, переопределяя метод compare(),

Collections.sort(unsortedList,new Comparator<Person>() {
    @Override
    public int compare(Person a, Person b) {
        return b.getName().compareTo(a.getName());
    }
});

Или используя Collection.reverse() для сортировки по убыванию как пользователь 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 выводятся как тип Person по выражению лямбда.

Ответ 2

Для чего стоит мой стандартный ответ. Единственное новое здесь - использование 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);
    }
}

Ответ 3

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]

Теперь вам нужно реализовать класс Comparator для вашего класса Person. Что-то вроде (для возрастания): 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)).

Чтобы свести к минимуму путаницу, вы должны реализовать восходящий компаратор и преобразовать его в нисходящий с оберткой (как это) new ReverseComparator<Person>(new PersonComparator()).

Ответ 4

Я бы создал компаратор для класса person, который может быть параметризован с определенным поведением сортировки. Здесь я могу установить порядок сортировки, но его можно изменить, чтобы разрешить сортировку и для других атрибутов 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 под рукой, закодирован "слепой" )

Edit

Благодаря Томасу, отредактировал код. Я бы не сказал, что использование Math.signum является хорошим, эффективным, эффективным, но я хотел бы сохранить его в качестве напоминания о том, что метод compareTo может возвращать любое целое число и умножение на (-1) не будет выполнено, если реализация возвращает Integer.MIN_INTEGER... И я удалил сеттер, потому что он достаточно дешев, чтобы построить новый PersonComparator только тогда, когда это было необходимо.

Но я держу бокс, потому что он показывает, что я полагаюсь на существующую реализацию Comparable. Возможно, что-то вроде Comparable<Integer> age1 = new Integer(person1.getAge());, но это выглядело слишком уродливым. Идея заключалась в том, чтобы показать шаблон, который можно легко адаптировать к другим атрибутам Person, таким как имя, день рождения, как Date и т.д.

Ответ 5

Использование Коллекций Google:

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());
}

Ответ 6

Класс java.util.Collection имеет метод сортировки, который принимает список и пользовательский Comparator. Вы можете определить свой собственный Компаратор, чтобы отсортировать свой объект Person, как вам нравится.

Ответ 7

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);
    }
}

}

Вот один из способов сделать это.