Sorting an ArrayList of objects using a custom sorting order

后端 未结 11 1455
说谎
说谎 2020-11-22 00:14

I am looking to implement a sort feature for my address book application.

I want to sort an ArrayList contactArray. Contact

相关标签:
11条回答
  • 2020-11-22 00:33

    BalusC and bguiz have already given very complete answers on how to use Java's built-in Comparators.

    I just want to add that google-collections has an Ordering class which is more "powerful" than the standard Comparators. It might be worth checking out. You can do cool things such as compounding Orderings, reversing them, ordering depending on a function's result for your objects...

    Here is a blog post that mentions some of its benefits.

    0 讨论(0)
  • 2020-11-22 00:35

    This page tells you all you need to know about sorting collections, such as ArrayList.

    Basically you need to

    • make your Contact class implement the Comparable interface by
      • creating a method public int compareTo(Contact anotherContact) within it.
    • Once you do this, you can just call Collections.sort(myContactList);,
      • where myContactList is ArrayList<Contact> (or any other collection of Contact).

    There's another way as well, involving creating a Comparator class, and you can read about that from the linked page as well.

    Example:

    public class Contact implements Comparable<Contact> {
    
        ....
    
        //return -1 for less than, 0 for equals, and 1 for more than
        public compareTo(Contact anotherContact) {
            int result = 0;
            result = getName().compareTo(anotherContact.getName());
            if (result != 0)
            {
                return result;
            }
            result = getNunmber().compareTo(anotherContact.getNumber());
            if (result != 0)
            {
                return result;
            }
            ...
        }
    }
    
    0 讨论(0)
  • 2020-11-22 00:37

    use this method:

    private ArrayList<myClass> sortList(ArrayList<myClass> list) {
        if (list != null && list.size() > 1) {
            Collections.sort(list, new Comparator<myClass>() {
                public int compare(myClass o1, myClass o2) {
                    if (o1.getsortnumber() == o2.getsortnumber()) return 0;
                    return o1.getsortnumber() < o2.getsortnumber() ? 1 : -1;
                }
            });
        }
        return list;
    }
    

    `

    and use: mySortedlist = sortList(myList); No need to implement comparator in your class. If you want inverse order swap 1 and -1

    0 讨论(0)
  • 2020-11-22 00:44

    Ok, I know this was answered a long time ago... but, here's some new info:

    Say the Contact class in question already has a defined natural ordering via implementing Comparable, but you want to override that ordering, say by name. Here's the modern way to do it:

    List<Contact> contacts = ...;
    
    contacts.sort(Comparator.comparing(Contact::getName).reversed().thenComparing(Comparator.naturalOrder());
    

    This way it will sort by name first (in reverse order), and then for name collisions it will fall back to the 'natural' ordering implemented by the Contact class itself.

    0 讨论(0)
  • 2020-11-22 00:50

    Here's a tutorial about ordering objects:

    • The Java Tutorials - Collections - Object Ordering

    Although I will give some examples, I would recommend to read it anyway.


    There are various way to sort an ArrayList. If you want to define a natural (default) ordering, then you need to let Contact implement Comparable. Assuming that you want to sort by default on name, then do (nullchecks omitted for simplicity):

    public class Contact implements Comparable<Contact> {
    
        private String name;
        private String phone;
        private Address address;
    
        public int compareTo(Contact other) {
            return name.compareTo(other.name);
        }
    
        // Add/generate getters/setters and other boilerplate.
    }
    

    so that you can just do

    List<Contact> contacts = new ArrayList<Contact>();
    // Fill it.
    
    Collections.sort(contacts);
    

    If you want to define an external controllable ordering (which overrides the natural ordering), then you need to create a Comparator:

    List<Contact> contacts = new ArrayList<Contact>();
    // Fill it.
    
    // Now sort by address instead of name (default).
    Collections.sort(contacts, new Comparator<Contact>() {
        public int compare(Contact one, Contact other) {
            return one.getAddress().compareTo(other.getAddress());
        }
    }); 
    

    You can even define the Comparators in the Contact itself so that you can reuse them instead of recreating them everytime:

    public class Contact {
    
        private String name;
        private String phone;
        private Address address;
    
        // ...
    
        public static Comparator<Contact> COMPARE_BY_PHONE = new Comparator<Contact>() {
            public int compare(Contact one, Contact other) {
                return one.phone.compareTo(other.phone);
            }
        };
    
        public static Comparator<Contact> COMPARE_BY_ADDRESS = new Comparator<Contact>() {
            public int compare(Contact one, Contact other) {
                return one.address.compareTo(other.address);
            }
        };
    
    }
    

    which can be used as follows:

    List<Contact> contacts = new ArrayList<Contact>();
    // Fill it.
    
    // Sort by address.
    Collections.sort(contacts, Contact.COMPARE_BY_ADDRESS);
    
    // Sort later by phone.
    Collections.sort(contacts, Contact.COMPARE_BY_PHONE);
    

    And to cream the top off, you could consider to use a generic javabean comparator:

    public class BeanComparator implements Comparator<Object> {
    
        private String getter;
    
        public BeanComparator(String field) {
            this.getter = "get" + field.substring(0, 1).toUpperCase() + field.substring(1);
        }
    
        public int compare(Object o1, Object o2) {
            try {
                if (o1 != null && o2 != null) {
                    o1 = o1.getClass().getMethod(getter, new Class[0]).invoke(o1, new Object[0]);
                    o2 = o2.getClass().getMethod(getter, new Class[0]).invoke(o2, new Object[0]);
                }
            } catch (Exception e) {
                // If this exception occurs, then it is usually a fault of the developer.
                throw new RuntimeException("Cannot compare " + o1 + " with " + o2 + " on " + getter, e);
            }
    
            return (o1 == null) ? -1 : ((o2 == null) ? 1 : ((Comparable<Object>) o1).compareTo(o2));
        }
    
    }
    

    which you can use as follows:

    // Sort on "phone" field of the Contact bean.
    Collections.sort(contacts, new BeanComparator("phone"));
    

    (as you see in the code, possibly null fields are already covered to avoid NPE's during sort)

    0 讨论(0)
  • 2020-11-22 00:50

    By using lambdaj you can sort a collection of your contacts (for example by their name) as it follows

    sort(contacts, on(Contact.class).getName());
    

    or by their address:

    sort(contacts, on(Contacts.class).getAddress());
    

    and so on. More in general, it offers a DSL to access and manipulate your collections in many ways, like filtering or grouping your contacts based on some conditions, aggregate some of their property values, etc.

    0 讨论(0)
提交回复
热议问题