国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Java > 正文

Java中自然排序和比較器排序詳解

2019-11-26 13:48:40
字體:
供稿:網(wǎng)友

前言

當(dāng)指執(zhí)行插入排序、希爾排序、歸并排序等算法時,比較兩個對象“大小”的比較操作。我們很容易理解整型的 i>j 這樣的比較方式,但當(dāng)我們對多個對象進行排序時,如何比較兩個對象的“大小”呢?這樣的比較 stu1 > stu2 顯然是不可能通過編譯的。為了解決如何比較兩個對象大小的問題,JDK提供了兩個接口 java.lang.Comparable java.util.Comparator

一、自然排序:java.lang.Comparable

Comparable 接口中只提供了一個方法: compareTo(Object obj) ,該方法的返回值是 int 。如果返回值為正數(shù),則表示當(dāng)前對象(調(diào)用該方法的對象)比 obj 對象“大”;反之“小”;如果為零的話,則表示兩對象相等。

下面是一個實現(xiàn)了 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 實現(xiàn)了自然排序接口 Comparable ,那么我們是怎么利用這個接口對一組 Student 對象進行排序的呢?我們在學(xué)習(xí)數(shù)組的時候,使用了一個類來給整型數(shù)組排序: java.util.Arrays 。我們使用 Arrays 的 sort 方法來給整型數(shù)組排序。翻翻 API 文檔就會發(fā)現(xiàn), Arrays 里給出了 sort 方法很多重載形式,其中就包括 sort(Object[] obj) ,也就是說 Arryas 也能對對象數(shù)組進行排序,排序過程中比較兩個對象“大小”時使用的就是 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 數(shù)組里添加元素的順序并不是按學(xué)號 id 來添加的。調(diào)用了 Arrays.sort(stus) 之后,對 Student 數(shù)組進行排序,不管 sort 是使用哪種排序算法來實現(xiàn)的,比較兩個對象“大小”這個操作,它是肯定要做的。那么如何比較兩個對象的“大小”? Student 實現(xiàn)的 Comparable 接口就發(fā)揮作用了。 sort 方法會將待比較的那個對象強制類型轉(zhuǎn)換成 Comparable ,并調(diào)用 compareTo 方法,根據(jù)其返回值來判斷這兩個對象的“大小”。所以,在這個例子中排序后的原 Student 亂序數(shù)組就變成了按學(xué)號排序的 Student 數(shù)組。

但是我們注意到,排序算法和 Student 類綁定了, Student 只有一種排序算法。但現(xiàn)實社會不是這樣的,如果我們不想按學(xué)號排序怎么辦?假如,我們想按姓名來給學(xué)生排序怎么辦?我們只能修改 Student 類的 Comparable 接口的 compareTo 方法,改成按姓名排序。如果在同一個系統(tǒng)里有兩個操作,一個是按學(xué)號排序,另外一個是按姓名排序,這怎么辦?不可能在 Student 類體中寫兩個 compareTo 方法的實現(xiàn)。這么看來Comparable就有局限性了。為了彌補這個不足,JDK 還為我們提供了另外一個排序方式,也就是下面要說的比較器排序。

二、比較器排序:java.util.Comparator

上面我提到了,之所以提供比較器排序接口,是因為有時需要對同一對象進行多種不同方式的排序,這點自然排序 Comparable 不能實現(xiàn)。另外, Comparator 接口的一個好處是將比較排序算法和具體的實體類分離了。

翻翻 API 會發(fā)現(xiàn), Arrays.sort 還有種重載形式:sort(T[] a, Comparator<? super T> c) ,這個方法參數(shù)的寫法用到了泛型,我們還沒講到。我們可以把它理解成這樣的形式: sort(Object[] a, Comparator c) ,這個方法的意思是按照比較器 c 給出的比較排序算法,對 Object 數(shù)組進行排序。Comparator 接口中定義了兩個方法: compare(Object o1, Object o2) equals 方法,由于 equals 方法所有對象都有的方法,因此當(dāng)我們實現(xiàn) Comparator 接口時,我們只需重寫 compare 方法,而不需重寫 equals 方法。Comparator 接口中對重寫 equals 方法的描述是:“注意,不重寫 Object.equals(Object) 方法總是安全的。然而,在某些情況下,重寫此方法可以允許程序確定兩個不同的 Comparator 是否強行實施了相同的排序,從而提高性能。”。我們只需知道第一句話就OK了,也就是說,可以不用去想應(yīng)該怎么實現(xiàn) equals 方法,因為即使我們不顯示實現(xiàn) equals 方法,而是使用Object類的 equals 方法,代碼依然是安全的。

那么我們來寫個代碼,來用一用比較器排序。還是用 Student 類來做,只是沒有實現(xiàn) Comparable 接口。由于比較器的實現(xiàn)類只用顯示實現(xiàn)一個方法,因此,我們可以不用專門寫一個類來實現(xiàn)它,當(dāng)我們需要用到比較器時,可以寫個匿名內(nèi)部類來實現(xiàn) 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)); } 

當(dāng)我們需要對Student按學(xué)號排序時,只需修改我們的排序方法中實現(xiàn)Comparator的內(nèi)部類中的代碼,而不用修改 Student 類。

注意: 當(dāng)然,你也可以用 Student 類實現(xiàn) Comparator 接口,這樣Student就是(is a)比較器了(Comparator)。當(dāng)需要使用這種排序的時候,將 Student 看作 Comparator 來使用就可以了,可以將 Student 作為參數(shù)傳入 sort 方法,因為 Student is a Comparator 。但這樣的代碼不是個優(yōu)秀的代碼,因為我們之所以使用比較器(Comparator),其中有個重要的原因就是,這樣可以把比較算法和具體類分離,降低類之間的耦合。

TreeSet對這兩種比較方式都提供了支持,分別對應(yīng)著TreeSet的兩個構(gòu)造方法:

     1、TreeSet():根據(jù)TreeSet中元素實現(xiàn)的 Comparable 接口的 compareTo 方法比較排序

   2、TreeSet(Comparator comparator):根據(jù)給定的 comparator 比較器,對 TreeSet 中的元素比較排序

當(dāng)向 TreeSet 中添加元素時,TreeSet 就會對元素進行排序。至于是用自然排序還是用比較器排序,就看你的 TreeSet 構(gòu)造是怎么寫的了。當(dāng)然,添加第一個元素時不會進行任何比較, TreeSet 中都沒有元素,和誰比去啊?

下面,分別給出使用兩種排序比較方式的 TreeSet 測試代碼:

/**  * 使用自然排序  * Student必須實現(xiàn)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沒有實現(xiàn)Comparable接口,拋出ClassCastException  set.add(stu4);  set.add(stu2);  set.add(stu4);  set.add(new Student(12, "Little"));   System.out.println(set); } 
/**  * 使用比較器排序  * Student可以只是個簡單的Java類,不用實現(xiàn)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提供了一系列用于對數(shù)組操作的靜態(tài)方法,查找排序等等。Collections也提供了一系列這樣的方法,只是它是用于處理集合的,雖然Collections類和Collection接口很像,但是不要被Collections的名字給欺騙了,它不是只能處理Collection接口以及子接口的實現(xiàn)類,同樣也可以處理Map接口的實現(xiàn)類。

總結(jié)

Java中自然排序和比較器排序的介紹就到這里了,文章介紹的還是相對詳細的,希望能對大家的學(xué)習(xí)或者工作帶來一定的幫助,如果有疑問大家可以留言交流。

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 静海县| 宝坻区| 大洼县| 辰溪县| 渝北区| 文昌市| 新竹县| 信丰县| 从化市| 北碚区| 夏邑县| 白河县| 泾川县| 噶尔县| 仪陇县| 泗阳县| 山东| 江安县| 平潭县| 绥德县| 上杭县| 金阳县| 永和县| 浑源县| 株洲县| 彰化市| 丰台区| 长白| 甘德县| 永年县| 南部县| 桂平市| 礼泉县| 竹山县| 芦溪县| 定边县| 和平区| 锡林郭勒盟| 高要市| 贡嘎县| 田东县|