it-swarm.com.ru

Сортировка коллекции объектов

Если у меня есть простой список строк:

List<String> stringList = new ArrayList<String>();

Я могу отсортировать это с:

Collections.sort(stringList);

Но предположим, что у меня есть класс Person:

public class Person
{
   private String name;
   private Integer age;
   private String country;
}

И список этого:

List<Person> personList = new ArrayList<Person>();

И я хочу отсортировать это иногда по имени, иногда по возрасту, иногда по стране.

Какой самый простой способ сделать это?

Я знаю, что могу реализовать интерфейс Comparable, но это, кажется, ограничивает меня в сортировке по одному конкретному свойству.

57
Knut Arne Vedaa

Реализуйте интерфейс Comparator (один раз для каждого другого порядка сортировки) и используйте метод Collections.sort () , который принимает Comparator в качестве дополнительного параметра.

34
Michael Borgwardt

Collections.sort можно вызывать с помощью специального компаратора. И этот компаратор может быть реализован для сортировки в разных порядках сортировки. Вот пример (для вашей модели Person - с возрастом в виде целого числа):

public class FlexiblePersonComparator implements Comparator<Person> {
  public enum Order {Name, Age, Country}

  private Order sortingBy = Name;

  @Override
  public int compare(Person person1, Person person2) {
    switch(sortingBy) {
      case Name: return person1.name.compareTo(person2.name);
      case Age: return person1.age.compareTo(person2.age);
      case Country: return person1.country.compareTo(person2.country);
    }
    throw new RuntimeException("Practically unreachable code, can't be thrown");
  }

  public void setSortingBy(Order sortBy) {
    this.sortingBy = sortingBy;
  }
}

И вы используете это так (при условии, что люди это поле):

public void sortPersonsBy(FlexiblePersonComparator.Order sortingBy) {
  List<Person> persons = this.persons;  // useless line, just for clarification
  FlexiblePersonComparator comparator = new FlexiblePersonComparator();
  comparator.setSortingBy(sortingBy);
  Collections.sort(persons, comparator); // now we have a sorted list
}
48
Andreas_D

Спасибо респондентам. Для блага других я хотел бы привести полный пример.

Решением является создание следующих дополнительных классов:

public class NameComparator implements Comparator<Person>
{
    public int compare(Person o1, Person o2)
    {
       return o1.getName().compareTo(o2.getName());
   }
}

public class AgeComparator implements Comparator<Person>
{
    public int compare(Person o1, Person o2)
    {
        return o1.getAge().compareTo(o2.getAge());
    }
}

public class CountryComparator implements Comparator<Person>
{
    public int compare(Person o1, Person o2)
    {
        return o1.getCountry().compareTo(o2.getCountry());
    }
}

Затем список можно отсортировать так:

Collections.sort(personList, new NameComparator());
Collections.sort(personList, new AgeComparator());
Collections.sort(personList, new CountryComparator());
17
Knut Arne Vedaa

Java 8 способ сделать это, используя List.sort следующим образом:

personList.sort(Comparator.comparing(Person::getName));

Цитировать Стюарт Маркс в своем ответе закончился здесь .

Это большое преимущество метода расширения List.sort(cmp) перед Collections.sort(list, cmp). Может показаться, что это всего лишь небольшое синтаксическое преимущество, заключающееся в возможности писать myList.sort(cmp) вместо Collections.sort(myList, cmp). Разница в том, что myList.sort(cmp), являясь методом расширения интерфейса, может быть переопределен конкретной реализацией List. Например, ArrayList.sort(cmp) сортирует список на месте с помощью Arrays.sort(), тогда как реализация по умолчанию реализует старый метод copyout-sort-copyback.

10
aioobe

Вы также можете использовать BeanComparator из Apache commons beanutils, например так:

Collections.sort(personList, new BeanComparator("name"));
6
Jörn Horstmann

Реализуйте 3 различных типа компаратора.

вы можете добавить компаратор в команду сортировки. Компаратор, который вы определяете, будет сортировать элементы по имени, возрасту или тому, что когда-либо.

Collections.sort(list, new Comparator() {

        public int compare(Object arg0, Object arg1) {
            if (!(arg0 instanceof Person)) {
                return -1;
            }
            if (!(arg1 instanceof Person)) {
                return -1;
            }

            Person pers0 = (Person)arg0;
            Person pers1 = (Person)arg1;


            // COMPARE NOW WHAT YOU WANT
            // Thanks to Steve Kuo for your comment!
            return pers0.getAge() - pers1.getAge();
        }
    });
3
Markus Lausberg

Метод Collections.sort может быть вызван со вторым аргументом, который является компаратором для использования . Создайте 3 компаратора и используйте тот, который вы хотите, когда это необходимо.

Collections.sort(list , new Comparator() {
        public int compare(Object o1, Object o2) {
          ...
        }
      });
2
z5h

Используя lambdaj ( http://code.google.com/p/lambdaj/ ), вы можете добиться того, что вы просите, следующим образом:

sort (personList, on (Person.class) .getName ());

sort (personList, on (Person.class) .getAge ());

sort (personList, on (Person.class) .getCountry ());

0
Mario Fusco

Я задал очень похожий вопрос (о поиске, а не сортировке), возможно, есть некоторая полезная информация (в итоге я использовал enum, который реализует Comparator, поэтому я передаю значение enum в качестве селектора компаратора).

0
Jason S