java संपत्ति द्वारा कस्टम ऑब्जेक्ट्स के ArrayList सॉर्ट करें




sorting date (20)

मैंने एक तुलनाकर्ता का उपयोग करके ArrayLists को सॉर्ट करने के बारे में पढ़ा है, लेकिन उदाहरण के लिए इस्तेमाल किए गए सभी उदाहरणों में जो कुछ शोध के अनुसार स्ट्रिंग्स के लिए एक विधि है।

मैं अपनी वस्तुओं में से एक द्वारा कस्टम ऑब्जेक्ट्स के getStartDay() को सॉर्ट करना चाहता था: एक डेट ऑब्जेक्ट ( getStartDay() )। आम तौर पर मैं उन्हें item1.getStartDate().before(item2.getStartDate()) तुलना करता हूं item1.getStartDate().before(item2.getStartDate()) तो मैं सोच रहा था कि मैं कुछ लिख सकता हूं:

public class CustomComparator {
    public boolean compare(Object object1, Object object2) {
        return object1.getStartDate().before(object2.getStartDate());
    }
}

public class RandomName {
    ...
    Collections.sort(Database.arrayList, new CustomComparator);
    ...
}

चूंकि Date Comparable से लागू होता है, इसकी compareTo एक String करता है जैसे String करता है।

तो आपका कस्टम Comparator इस तरह दिख सकता है:

public class CustomComparator implements Comparator<MyObject> {
    @Override
    public int compare(MyObject o1, MyObject o2) {
        return o1.getStartDate().compareTo(o2.getStartDate());
    }
}

compare() विधि को एक int वापस करना होगा, ताकि आप सीधे boolean वापस नहीं कर सकें जैसे कि आप वैसे भी योजना बना रहे थे।

आपका सॉर्टिंग कोड आपके जैसा लिखा होगा:

Collections.sort(Database.arrayList, new CustomComparator());

यह सब लिखने का थोड़ा सा तरीका है, अगर आपको अपने तुलनित्र का पुन: उपयोग करने की आवश्यकता नहीं है, तो इसे एक इनलाइन अज्ञात वर्ग के रूप में लिखना है:

Collections.sort(Database.arrayList, new Comparator<MyObject>() {
    @Override
    public int compare(MyObject o1, MyObject o2) {
        return o1.getStartDate().compareTo(o2.getStartDate());
    }
});

java-8 बाद से

अब आप Comparator लिए लैम्ब्डा अभिव्यक्ति का उपयोग कर एक छोटे से रूप में अंतिम उदाहरण लिख सकते हैं:

Collections.sort(Database.arrayList, 
                        (o1, o2) -> o1.getStartDate().compareTo(o2.getStartDate()));

और List में एक sort(Comparator) विधि है, इसलिए आप इसे और भी कम कर सकते हैं:

Database.arrayList.sort((o1, o2) -> o1.getStartDate().compareTo(o2.getStartDate()));

यह एक सामान्य मुहावरे है कि एक Comparable कुंजी वाले वर्ग के लिए Comparator उत्पन्न करने के लिए एक अंतर्निहित विधि है:

Database.arrayList.sort(Comparator.comparing(MyObject::getStartDate));

ये सभी समकक्ष रूप हैं।


Java 8 से आगे और आगे हमें Collections.sort() का उपयोग करने की आवश्यकता नहीं है। List इंटरफ़ेस में एक डिफ़ॉल्ट sort() विधि है:

List<User> users = Arrays.asList(user1,user2,user3);
users.sort( (u1, u2) -> { 
return u1.getFirstName.compareTo(u2.getFirstName());}); 

http://visvv.blogspot.in/2016/01/sorting-objects-in-java-8.html देखें।


जिन वर्गों में एक प्राकृतिक क्रम क्रम है (एक वर्ग संख्या, उदाहरण के रूप में) को तुलनात्मक इंटरफ़ेस को कार्यान्वित करना चाहिए, जबकि कक्षाओं में कोई प्राकृतिक सॉर्ट ऑर्डर नहीं है (एक वर्ग अध्यक्ष, उदाहरण के रूप में) एक तुलनात्मक (या एक अनाम तुलनाकर्ता) के साथ प्रदान किया जाना चाहिए कक्षा)।

दो उदाहरण:

public class Number implements Comparable<Number> {
    private int value;

    public Number(int value) { this.value = value; }
    public int compareTo(Number anotherInstance) {
        return this.value - anotherInstance.value;
    }
}

public class Chair {
    private int weight;
    private int height;

    public Chair(int weight, int height) {
        this.weight = weight;
        this.height = height;
    }
    /* Omitting getters and setters */
}
class ChairWeightComparator implements Comparator<Chair> {
    public int compare(Chair chair1, Chair chair2) {
        return chair1.getWeight() - chair2.getWeight();
    }
}
class ChairHeightComparator implements Comparator<Chair> {
    public int compare(Chair chair1, Chair chair2) {
        return chair1.getHeight() - chair2.getHeight();
    }
}

उपयोग:

List<Number> numbers = new ArrayList<Number>();
...
Collections.sort(numbers);

List<Chair> chairs = new ArrayList<Chair>();
// Sort by weight:
Collections.sort(chairs, new ChairWeightComparator());
// Sort by height:
Collections.sort(chairs, new ChairHeightComparator());

// You can also create anonymous comparators;
// Sort by color:
Collections.sort(chairs, new Comparator<Chair>() {
    public int compare(Chair chair1, Chair chair2) {
        ...
    }
});

मैं इस प्रक्रिया को पसंद करता हूं:

public class SortUtil
{    
    public static <T> List<T> sort(List<T> list, String sortByProperty)
    {
            Collections.sort(list, new BeanComparator(sortByProperty));
            return list;
    }
}

List<T> sortedList = SortUtil<T>.sort(unsortedList, "startDate");

यदि आप ऑब्जेक्ट्स की सूची में startDate नामक एक संपत्ति है, तो आप इसे ओवर-ओवर का उपयोग करते हैं। आप उन्हें startDate.time भी शुरू कर सकते हैं।

इसके लिए आपकी ऑब्जेक्ट को Comparable होने की आवश्यकता है जिसका अर्थ है कि आपको compareTo करने की आवश्यकता है, equals , और hashCode कार्यान्वयन।

हां, यह तेज़ हो सकता है ... लेकिन अब आपको प्रत्येक प्रकार के प्रकार के लिए एक नया तुलना करने की ज़रूरत नहीं है। यदि आप देव समय पर बचा सकते हैं और रनटाइम पर छोड़ सकते हैं, तो आप इसके साथ जा सकते हैं।


जावा 8 लैम्ब्डा इस प्रकार को छोटा करता है।

Collections.sort(stdList, (o1, o2) -> o1.getName().compareTo(o2.getName()));

मुझे सबसे अधिक पता चला है कि अगर ये सभी उत्तर तुलनात्मक लागू करने के लिए अंतर्निहित वर्ग (ऑब्जेक्ट) पर निर्भर नहीं हैं या एक सहायक तुलनात्मक इंटरफ़ेस है।

मेरे समाधान के साथ नहीं! निम्न कोड आपको स्ट्रिंग नाम जानने के द्वारा ऑब्जेक्ट के फ़ील्ड की तुलना करने देता है। आप इसे आसानी से नाम का उपयोग न करने के लिए संशोधित कर सकते हैं, लेकिन फिर आपको इसे बेनकाब करने या उन ऑब्जेक्ट्स में से एक बनाने की आवश्यकता है, जिनके साथ आप तुलना करना चाहते हैं।

Collections.sort(anArrayListOfSomeObjectPerhapsUsersOrSomething, new ReflectiveComparator(). new ListComparator("name"));

public class ReflectiveComparator {
    public class FieldComparator implements Comparator<Object> {
        private String fieldName;

        public FieldComparator(String fieldName){
            this.fieldName = fieldName;
        }

        @SuppressWarnings({ "unchecked", "rawtypes" })
        @Override
        public int compare(Object object1, Object object2) {
            try {
                Field field = object1.getClass().getDeclaredField(fieldName);
                field.setAccessible(true);

                Comparable object1FieldValue = (Comparable) field.get(object1);
                Comparable object2FieldValue = (Comparable) field.get(object2);

                return object1FieldValue.compareTo(object2FieldValue);
            }catch (Exception e){}

            return 0;
        }
    }

    public class ListComparator implements Comparator<Object> {
        private String fieldName;

        public ListComparator(String fieldName) {
            this.fieldName = fieldName;
        }

        @SuppressWarnings({ "unchecked", "rawtypes" })
        @Override
        public int compare(Object object1, Object object2) {
            try {
                Field field = object1.getClass().getDeclaredField(fieldName);
                field.setAccessible(true);
                Comparable o1FieldValue = (Comparable) field.get(object1);
                Comparable o2FieldValue = (Comparable) field.get(object2);

                if (o1FieldValue == null){ return -1;}
                if (o2FieldValue == null){ return 1;}
                return o1FieldValue.compareTo(o2FieldValue);
            } catch (NoSuchFieldException e) {
                throw new IllegalStateException("Field doesn't exist", e);
            } catch (IllegalAccessException e) {
                throw new IllegalStateException("Field inaccessible", e);
            }
        }
    }
}

आपके कस्टमकंपेटर वर्ग को उपयोग करने के लिए java.util.Comparator को लागू करना होगा। इसे भी तुलना करना चाहिए () और बराबर ()

तुलना करें () को प्रश्न का उत्तर देना चाहिए: क्या ऑब्जेक्ट 1 ऑब्जेक्ट 2 के बराबर या उससे अधिक है?

पूर्ण दस्तावेज़: http://java.sun.com/j2se/1.5.0/docs/api/java/util/Comparator.html


जावा 8 के लिए:

Collections.sort(list, comparing(ClassName::getName));

या

Collections.sort(list, comparing(ClassName::getName).reversed());

ऐसा लगता है कि स्ट्रिंग दिनांक प्रकार के साथ "2015-12-14T21: 55: 51Z"

एक और तरीका है

Collections.sort(list, comparing(ClassName::getName, Comparator.nullsLast(Comparator.naturalOrder())));

आप अमरूद Ordering प्रयास कर सकते हैं:

Function<Item, Date> getStartDate = new Function<Item, Date>() {
    public Date apply(Item item) {
        return item.getStartDate();
    }
};

List<Item> orderedItems = Ordering.natural().onResultOf(getStartDate).
                          sortedCopy(items);

आप जावा 8 का उपयोग कर सॉर्ट कर सकते हैं

yourList.sort(Comparator.comparing(Classname::getName));

or

yourList.stream().forEach(a -> a.getBObjects().sort(Comparator.comparing(Classname::getValue)));

इस पुस्तकालय के साथ आप कई कॉलम पर कस्टम ऑब्जेक्ट्स की सूची को सॉर्ट कर सकते हैं। पुस्तकालय संस्करण 8.0 सुविधाओं का उपयोग करता है। नमूना भी उपलब्ध है। यहां करने के लिए एक नमूना है

SortKeys sortKeys = new SortKeys();
sortKeys.addField("firstName")
            .addField("age", true); // This (true) will sort the age descending

// Other ways to specify a property to the sorter are
//      .addField("lastName", String.class);
//      .addField("dob", Date.class, true);

// Instantiate a ListSorter
ListSorter listSorter = new ListSorter();

// Pass the data to sort (listToSort) and the "by keys" to sort (sortKeys)
List sortedList = (List<Person>) listSorter.sortList(listToSort, sortKeys);

हाँ तुम कर सकते हो। वस्तुओं, Comparable इंटरफ़ेस और Comparable इंटरफ़ेस की तुलना करने के दो विकल्प हैं।

इन दोनों इंटरफेस अलग-अलग व्यवहार की अनुमति देते हैं। तुलनात्मक आपको ऑब्जेक्ट एक्ट करने की इजाजत देता है जैसे आपने स्ट्रिंग्स का वर्णन किया है (वास्तव में, स्ट्रिंग लागू तुलनात्मक)। दूसरा, तुलनात्मक, आपको वह करने की अनुमति देता है जो आप करना चाहते हैं। आप इसे इस तरह करेंगे:

Collections.sort(myArrayList, new MyComparator());

इससे Collectionssort विधि आपके सॉर्टिंग तंत्र के लिए आपके तुलनित्र का उपयोग करने का कारण बन जाएगी। यदि ArrayList में ऑब्जेक्ट तुलनीय लागू करते हैं, तो आप इसके बजाय कुछ ऐसा कर सकते हैं:

Collections.sort(myArrayList);

Collections क्लास में इनमें से कई उपयोगी, सामान्य उपकरण शामिल हैं।


जावा 8 लैम्ब्डा अभिव्यक्ति

Collections.sort(studList, (Student s1, Student s2) ->{
        return s1.getFirstName().compareToIgnoreCase(s2.getFirstName());
});

या

Comparator<Student> c = (s1, s2) -> s1.firstName.compareTo(s2.firstName);
studList.sort(c)

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.Date;

public class test {

public static class Person {
    public String name;
    public int id;
    public Date hireDate;

    public Person(String iname, int iid, Date ihireDate) {
        name = iname;
        id = iid;
        hireDate = ihireDate;
    }

    public String toString() {
        return name + " " + id + " " + hireDate.toString();
    }

    // Comparator
    public static class CompId implements Comparator<Person> {
        @Override
        public int compare(Person arg0, Person arg1) {
            return arg0.id - arg1.id;
        }
    }

    public static class CompDate implements Comparator<Person> {
        private int mod = 1;
        public CompDate(boolean desc) {
            if (desc) mod =-1;
        }
        @Override
        public int compare(Person arg0, Person arg1) {
            return mod*arg0.hireDate.compareTo(arg1.hireDate);
        }
    }
}

public static void main(String[] args) {
    // TODO Auto-generated method stub
    SimpleDateFormat df = new SimpleDateFormat("mm-dd-yyyy");
    ArrayList<Person> people;
    people = new ArrayList<Person>();
    try {
        people.add(new Person("Joe", 92422, df.parse("12-12-2010")));
        people.add(new Person("Joef", 24122, df.parse("1-12-2010")));
        people.add(new Person("Joee", 24922, df.parse("12-2-2010")));
    } catch (ParseException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }

    Collections.sort(people, new Person.CompId());
    System.out.println("BY ID");
    for (Person p : people) {
        System.out.println(p.toString());
    }

    Collections.sort(people, new Person.CompDate(false));
    System.out.println("BY Date asc");
    for (Person p : people) {
        System.out.println(p.toString());
    }
    Collections.sort(people, new Person.CompDate(true));
    System.out.println("BY Date desc");
    for (Person p : people) {
        System.out.println(p.toString());
    }

}

}

चूंकि प्रौद्योगिकियां हर रोज दिखाई देती हैं, इसलिए उस समय जवाब बदल जाएगा। मैंने LambdaJ पर एक नज़र डाली और बहुत दिलचस्प लगता है।

आप LambdaJ के साथ इन कार्यों को हल करने का प्रयास कर सकते हैं। आप इसे यहां पा सकते हैं: http://code.google.com/p/lambdaj/

यहां आपके पास एक उदाहरण है:

क्रमबद्ध क्रमबद्ध करें

List<Person> sortedByAgePersons = new ArrayList<Person>(persons);
Collections.sort(sortedByAgePersons, new Comparator<Person>() {
        public int compare(Person p1, Person p2) {
           return Integer.valueOf(p1.getAge()).compareTo(p2.getAge());
        }
});

लैम्ब्डा के साथ क्रमबद्ध करें

List<Person> sortedByAgePersons = sort(persons, on(Person.class).getAge()); 

बेशक, इस तरह की सुंदरता प्रदर्शन में प्रभाव डालती है (2 बार औसत), लेकिन क्या आप एक और अधिक पठनीय कोड पा सकते हैं?


जावा 8 के साथ सबसे आसान तरीका अंग्रेजी वर्णमाला के लिए है

कक्षा कार्यान्वयन

public class NewspaperClass implements Comparable<NewspaperClass>{
   public String name;

   @Override
   public int compareTo(NewspaperClass another) {
      return name.compareTo(another.name);
   }
}

तरह

  Collections.sort(Your List);

यदि आप वर्णमाला के लिए सॉर्ट करना चाहते हैं जिसमें गैर अंग्रेजी वर्ण हैं तो आप लोकेल का उपयोग कर सकते हैं ... कोड के नीचे तुर्की चरित्र प्रकार का उपयोग करें ...

कक्षा कार्यान्वयन

public class NewspaperClass implements Comparator<NewspaperClass> {
   public String name;
   public Boolean isUserNewspaper=false;
   private Collator trCollator = Collator.getInstance(new Locale("tr_TR"));



   @Override
   public int compare(NewspaperClass lhs, NewspaperClass rhs) {
      trCollator.setStrength(Collator.PRIMARY);
      return trCollator.compare(lhs.name,rhs.name);
   }
}

तरह

Collections.sort(your array list,new NewspaperClass());

एक ArrayList सॉर्ट करने के लिए आप निम्न कोड स्निपेट का उपयोग कर सकते हैं:

Collections.sort(studList, new Comparator<Student>(){
    public int compare(Student s1, Student s2) {
        return s1.getFirstName().compareToIgnoreCase(s2.getFirstName());
    }
});

2016 में स्टटगार्ट जर्मनी में जावा फोरम में आप इस presentation को देख सकते हैं।

केवल कुछ स्लाइड जर्मन भाषा का उपयोग करती हैं, 99% सामग्री "अंग्रेजी आधारित" जावा स्रोत कोड है; पसंद

someCollection.sort(
  OurCustomComparator
    .comparing(Person::getName)
    .thenComparing(Person::getId)
);

जहां OurCustomComparator डिफ़ॉल्ट विधियों (और अन्य रोचक विचारों) का उपयोग कर रहा है। जैसा कि दिखाया गया है, क्रमबद्ध करने के लिए कुछ गेटर विधि चुनने के लिए बहुत संक्षिप्त कोड का कारण बनता है; और सॉर्ट मानदंडों के सुपर सरल चेनिंग (या रिवर्सिंग)।

यदि आप जावा 8 में हैं, तो आपको शुरू करने के लिए वहां बहुत सारी सामग्री मिलती है।


Collection.sort () का उपयोग करने के बजाय 1.8 के बाद से एक list.sort () विधि है ताकि आप सीधे mylistcontainer.sort () को कॉल कर सकें

यहां एक कोड स्निपेट है जो List.sort () सुविधा प्रदर्शित करता है:

List<Fruit> fruits = new ArrayList<Fruit>();
fruits.add(new Fruit("Kiwi","green",40));
fruits.add(new Fruit("Banana","yellow",100));
fruits.add(new Fruit("Apple","mixed green,red",120));
fruits.add(new Fruit("Cherry","red",10));

// a) using an existing compareto() method
fruits.sort((Fruit f1,Fruit f2) -> f1.getFruitName().compareTo(f2.getFruitName()));
System.out.println("Using String.compareTo(): " + fruits);
//Using String.compareTo(): [Apple is: mixed green,red, Banana is: yellow, Cherry is: red, Kiwi is: green]

// b) Using a comparable class
fruits.sort((Fruit f1,Fruit f2) -> f1.compareTo(f2));  
System.out.println("Using a Comparable Fruit class (sort by color): " + fruits);
// Using a Comparable Fruit class (sort by color): [Kiwi is green, Apple is: mixed green,red, Cherry is: red, Banana is: yellow]

फल वर्ग है:

public class Fruit implements Comparable<Fruit>
{
private String name;
private String color;
private int quantity;

public Fruit(String name,String color,int quantity)
{ this.name = name; this.color = color; this.quantity = quantity;}

public String getFruitName() { return name; }        
public String getColor() { return color; }  
public int getQuantity() { return quantity; }

@Override public final int compareTo(Fruit f) // sorting the color
{
    return this.color.compareTo(f.color);
}     
@Override public String toString()
{   return (name + " is: " + color);
}

} // फल वर्ग के अंत


हां, उदाहरण के लिए यह संभव है कि मैं इस श्रेणी में IndexValue के प्रॉपर्टी v द्वारा सॉर्ट करता IndexValue

    // Sorting by property v using a custom comparator.
    Arrays.sort( array, new Comparator<IndexValue>(){
        public int compare( IndexValue a, IndexValue b ){
            return a.v - b.v;
        }
    });

यदि आप यहां देखते हैं तो मैं एक अज्ञात भीतरी कक्षा बना रहा हूं (जो बंद करने के लिए जावा है) और इसे क्लास Arrays के sort विधि पर सीधे पास कर रहा है

आपकी ऑब्जेक्ट Comparable भी कार्यान्वित कर सकती है (यही वह है जो स्ट्रिंग और जावा में अधिकांश कोर लाइब्रेरी करता है) लेकिन यह कक्षा के "प्राकृतिक क्रम क्रम" को परिभाषित करेगा, और आपको नए प्लग करने की अनुमति नहीं देता है।





comparator