前言
當指執行插入排序、希爾排序、歸併排序等算法時,比較兩個對象“大小”的比較操作。我們很容易理解整型的i>j 這樣的比較方式,但當我們對多個對象進行排序時,如何比較兩個對象的“大小”呢?這樣的比較stu1 > stu2 顯然是不可能通過編譯的。為了解決如何比較兩個對像大小的問題,JDK提供了兩個接口java.lang.Comparable和java.util.Comparator 。
一、自然排序:java.lang.Comparable
Comparable 接口中只提供了一個方法: compareTo(Object obj) ,該方法的返回值是int 。如果返回值為正數,則表示當前對象(調用該方法的對象)比obj 對象“大”;反之“小”;如果為零的話,則表示兩對象相等。
下面是一個實現了Comparable 接口的Student 類:
public class Student implements Comparable { private int id; private String name; public Student() { super(); } @Override public int compareTo(Object obj) { if (obj instanceof Student) { Student stu = (Student) obj; return id - stu.id; } return 0; } @Override public String toString() { return "<" + id + ", " + name + ">"; } } Student 實現了自然排序接口Comparable ,那麼我們是怎麼利用這個接口對一組Student 對象進行排序的呢?我們在學習數組的時候,使用了一個類來給整型數組排序: java.util.Arrays 。我們使用Arrays 的sort 方法來給整型數組排序。翻翻API 文檔就會發現, Arrays 裡給出了sort 方法很多重載形式,其中就包括sort(Object[] obj) ,也就是說Arryas 也能對對像數組進行排序,排序過程中比較兩個對象“大小”時使用的就是Comparable 接口的compareTo 方法。
public class CompareTest { public static void main(String[] args) { Student stu1 = new Student(1, "Little"); Student stu2 = new Student(2, "Cyntin"); Student stu3 = new Student(3, "Tony"); Student stu4 = new Student(4, "Gemini"); Student[] stus = new Student[4]; stus[0] = stu1; stus[1] = stu4; stus[2] = stu3; stus[3] = stu2; System.out.println(“Array: ” + Arrays.toString(stus)); Arrays.sort(stus); System.out.println(“Sort: ” + Arrays.toString(stus)); } } Student 數組裡添加元素的順序並不是按學號id 來添加的。調用了Arrays.sort(stus)之後,對Student 數組進行排序,不管sort 是使用哪種排序算法來實現的,比較兩個對象“大小”這個操作,它是肯定要做的。那麼如何比較兩個對象的“大小”? Student 實現的Comparable 接口就發揮作用了。 sort 方法會將待比較的那個對象強制類型轉換成Comparable ,並調用compareTo 方法,根據其返回值來判斷這兩個對象的“大小”。所以,在這個例子中排序後的原Student 亂序數組就變成了按學號排序的Student 數組。
但是我們注意到,排序算法和Student 類綁定了, Student 只有一種排序算法。但現實社會不是這樣的,如果我們不想按學號排序怎麼辦?假如,我們想按姓名來給學生排序怎麼辦?我們只能修改Student 類的Comparable 接口的compareTo 方法,改成按姓名排序。如果在同一個系統裡有兩個操作,一個是按學號排序,另外一個是按姓名排序,這怎麼辦?不可能在Student 類體中寫兩個compareTo 方法的實現。這麼看來Comparable就有局限性了。為了彌補這個不足,JDK 還為我們提供了另外一個排序方式,也就是下面要說的比較器排序。
二、比較器排序:java.util.Comparator
上面我提到了,之所以提供比較器排序接口,是因為有時需要對同一對象進行多種不同方式的排序,這點自然排序Comparable 不能實現。另外, Comparator 接口的一個好處是將比較排序算法和具體的實體類分離了。
翻翻API 會發現, Arrays.sort 還有種重載形式: sort(T[] a, Comparator<? super T> c) ,這個方法參數的寫法用到了泛型,我們還沒講到。我們可以把它理解成這樣的形式: sort(Object[] a, Comparator c) ,這個方法的意思是按照比較器c 給出的比較排序算法,對Object 數組進行排序。 Comparator 接口中定義了兩個方法: compare(Object o1, Object o2)和equals方法,由於equals方法所有對像都有的方法,因此當我們實現Comparator 接口時,我們只需重寫compare方法,而不需重寫equals方法。 Comparator 接口中對重寫equals 方法的描述是:“注意,不重寫Object.equals(Object)方法總是安全的。然而,在某些情況下,重寫此方法可以允許程序確定兩個不同的Comparator 是否強行實施了相同的排序,從而提高性能。”。我們只需知道第一句話就OK了,也就是說,可以不用去想應該怎麼實現equals 方法,因為即使我們不顯示實現equals 方法,而是使用Object類的equals 方法,代碼依然是安全的。
那麼我們來寫個代碼,來用一用比較器排序。還是用Student 類來做,只是沒有實現Comparable 接口。由於比較器的實現類只用顯示實現一個方法,因此,我們可以不用專門寫一個類來實現它,當我們需要用到比較器時,可以寫個匿名內部類來實現Comparator 。
下面是我們的按姓名排序的方法:
public void sortByName () { Student stu1 = new Student(1, "Little"); Student stu2 = new Student(2, "Cyntin"); Student stu3 = new Student(3, "Tony"); Student stu4 = new Student(4, "Gemini"); Student[] stus = new Student[4]; stus[0] = stu1; stus[1] = stu4; stus[2] = stu3; stus[3] = stu2; System.out.println("Array: " + Arrays.toString(stus)); Arrays.sort(stus, new Comparator() { @Override public int compare(Object o1, Object o2) { if (o1 instanceof Student && o2 instanceof Student) { Student s1 = (Student) o1; Student s2 = (Student) o2; //return s1.getId() - s2.getId(); // 按Id排return s1.getName().compareTo(s2.getName()); // 按姓名排} return 0; } }); System.out.println("Sorted: " + Arrays.toString(stus)); }當我們需要對Student按學號排序時,只需修改我們的排序方法中實現Comparator的內部類中的代碼,而不用修改Student 類。
注意:當然,你也可以用Student 類實現Comparator 接口,這樣Student就是(is a)比較器了(Comparator)。當需要使用這種排序的時候,將Student 看作Comparator 來使用就可以了,可以將Student 作為參數傳入sort 方法,因為Student is a Comparator 。但這樣的代碼不是個優秀的代碼,因為我們之所以使用比較器(Comparator),其中有個重要的原因就是,這樣可以把比較算法和具體類分離,降低類之間的耦合。
TreeSet對這兩種比較方式都提供了支持,分別對應著TreeSet的兩個構造方法:
1、TreeSet():根據TreeSet中元素實現的Comparable 接口的compareTo 方法比較排序
2、TreeSet(Comparator comparator):根據給定的comparator 比較器,對TreeSet 中的元素比較排序
當向TreeSet 中添加元素時,TreeSet 就會對元素進行排序。至於是用自然排序還是用比較器排序,就看你的TreeSet 構造是怎麼寫的了。當然,添加第一個元素時不會進行任何比較, TreeSet 中都沒有元素,和誰比去啊?
下面,分別給出使用兩種排序比較方式的TreeSet 測試代碼:
/** * 使用自然排序* Student必須實現Comparable接口,否則會拋出ClassCastException */ public void testSortedSet3() { Student stu1 = new Student(1, "Little"); Student stu2 = new Student(2, "Cyntin"); Student stu3 = new Student(3, "Tony"); Student stu4 = new Student(4, "Gemini"); SortedSet set = new TreeSet(); set.add(stu1); set.add(stu3); // 若Student沒有實現Comparable接口,拋出ClassCastException set.add(stu4); set.add(stu2); set.add(stu4); set.add(new Student(12, "Little")); System.out.println(set); } /** * 使用比較器排序* Student可以只是個簡單的Java類,不用實現Comparable接口*/ public void testSortedSet3() { Student stu1 = new Student(1, "Little"); Student stu2 = new Student(2, "Cyntin"); Student stu3 = new Student(3, "Tony"); Student stu4 = new Student(4, "Gemini"); SortedSet set = new TreeSet(new Comparator() { @Override public int compare(Object o1, Object o2) { if (o1 instanceof Student && o2 instanceof Student) { Student s1 = (Student) o1; Student s2 = (Student) o2; return s1.getName().compareTo(s2.getName()); } return 0; } }); set.add(stu1); set.add(stu3); set.add(stu4); set.add(stu2); set.add(stu4); set.add(new Student(12, "Little")); System.out.println(set); }另外,介紹個工具類, java.util.Collections 。注意,這不是Collection接口。 Collections很像Arrays類。 Arrays提供了一系列用於對數組操作的靜態方法,查找排序等等。 Collections也提供了一系列這樣的方法,只是它是用於處理集合的,雖然Collections類和Collection接口很像,但是不要被Collections的名字給欺騙了,它不是只能處理Collection接口以及子接口的實現類,同樣也可以處理Map接口的實現類。
總結
Java中自然排序和比較器排序的介紹就到這裡了,文章介紹的還是相對詳細的,希望能對大家的學習或者工作帶來一定的幫助,如果有疑問大家可以留言交流。