【Java】Comparable & Comparator

今天自己翻了一下前面學(xué)習(xí)的知識,突然間看到這個話題玛臂,回想了一下烤蜕,感覺自己對Comparable和Comparator的理解印象不是很深了。就打算寫一下這一篇博客迹冤,重溫一下好了讽营。之后自己再不明確的時候,也可以來翻一下泡徙。


區(qū)別

  • Comparable & Comparator 都是用來實現(xiàn)集合中的元素比較橱鹏、排序的。其中Comparable是在集合內(nèi)部定義的方法實現(xiàn)的排序堪藐,Comparator是在集合外部絲線的排序莉兰。所以,要想實現(xiàn)排序的功能礁竞,就需要在集合內(nèi)實現(xiàn)Comparable接口的方法(也就是重寫compareTo()方法)或者在集合外實現(xiàn)Comparator接口的方法(也就是重寫compare()方法)糖荒。

  • Comparable位于包java.lang下,Comparator位于包java.util下模捂。

  • Comparable是一個類本身就已經(jīng)支持比較所需要實現(xiàn)的接口(如String捶朵、Integer自己就可以完成比較大小的操作蜘矢,因為已經(jīng)實現(xiàn)了Comparable接口)。自定義的類需要實現(xiàn)Comparable并把實例對象加入List集合中后才可以排序综看。而Comparator硼端,我們可以稱之為比較器。當(dāng)這個類不支持自身比較或者自比較的方法不能滿足我的要求的時候寓搬,我們可以寫一個比較器來完成兩個對象之間的大小比較珍昨。可以說一個是自己完成比較句喷,另外一個是通過外部的來程序完成比較镣典。

用Comparator是策略模式(Strategy design pattern),就是不改變對象自身唾琼,而用一個策略對象(strategy object)來改變他的行為兄春。

那就有一個疑問了,為什么需要存在Comparator呢锡溯?我們都用Comparable不就好了嗎赶舆?哈哈。這里我們就有一個場景啦祭饭。比如對于字符串的排序芜茵,String類已經(jīng)實現(xiàn)了Comparable接口,它的compareTo()方法是把字符串按照字典順序進行排列倡蝙。那我們?nèi)绻胍址凑臻L度來排序呢九串?String類是不允許我們修改它而且是一個final類(最終類)是不能被繼承的,也就是說我們不能去重寫compareTo()方法寺鸥。那么我們就必須實現(xiàn)Comparator接口猪钮,生成一個比較器。這樣的解釋應(yīng)該挺明白了吧胆建。

Comparable示例程序

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

public class StudentComparable implements Comparable<StudentComparable> {    
  private String name;    
  private int age;    

  public StudentComparable(String name, int age) {        
    this.name = name;        
    this.age = age;    
  }    

  public void setName(String name) {        
    this.name = name;    
  }    

  public void setAge(int age) {        
    this.age = age;    
  }    

  public String getName() {        
    return name;    
  }    

  public int getAge() {        
    return age;    
  }    

  @Override    
  public boolean equals(Object o) {        
    if (this == o) {
      return true;
    }
    if (o == null || getClass() != o.getClass()) {
      return false;
    }        

    StudentComparable that = (StudentComparable) o;        
    if (getAge() != that.getAge()) {
      return false;
    }        

    return getName() != null ? getName().equals(that.getName()) : that.getName() == null;    
  }    

  @Override    
  public int hashCode() {        
    int result = getName() != null ? getName().hashCode() : 0;        
    result = 31 * result + getAge();        
    return result;    
  }    

  @Override    
  public int compareTo(StudentComparable o) {        
    if (this.name.compareTo(o.getName()) > 0) {            
      return 1;        
    }else if (this.name.compareTo(o.getName()) < 0) {            
      return -1;        
    }        
    return 0;    
  }    

  @Override    
  public String toString() {        
    return "StudentComparable{" + "name='" + name + '\'' + ", age=" + age + '}';    
  }    

  public static void main(String[] args) {        
    List<StudentComparable> list = new ArrayList<>();        
    list.add(new StudentComparable("Jack", 16));        
    list.add(new StudentComparable("Mike", 25));        
    list.add(new StudentComparable("Summer", 17));        
    list.add(new StudentComparable("Bob", 19));        
    Collections.sort(list);        
    for (StudentComparable stu : list) {            
      System.out.println(stu);        
    }    
  }
}

Comparator示例代碼

import java.util.*;

class A implements Comparator<String> {    
  @Override    
  public int compare(String s1, String s2) {        
    int len1 = s1.length();        
    int len2 = s2.length();        
    if (len1 > len2) {            
      return 1;        
    }else if (len1 < len2) {            
      return -1;        
    }else {            
      return 0;        
    }    
  }
}

public class ComparatorTest {    
  public static void main(String[] args) {        
    List<String> list = new ArrayList<>();        
    list.add("cat");        
    list.add("fish");        
    list.add("dog");        
    list.add("panda");        
    Collections.sort(list, new A());        
    System.out.println(list);    
  }
}

注意

  • 在Comparable接口的實現(xiàn)類里烤低,我們要必須重寫compareTo()方法,看需求是否需要重寫hashCode()方法和equals()方法 (有關(guān)hashCode()和equals()的意義和用法笆载,詳見我的另一篇文章(待定))扑馁。

  • 在Comparator接口的實現(xiàn)類里,我們必須重寫compare()方法宰译,看需求是否需要重寫hashCode()方法和equals()方法 檐蚜。

  • 其實上述的hashCode()方法和equals()方法不重寫也是可以的。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末沿侈,一起剝皮案震驚了整個濱河市闯第,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌缀拭,老刑警劉巖咳短,帶你破解...
    沈念sama閱讀 219,490評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件填帽,死亡現(xiàn)場離奇詭異,居然都是意外死亡咙好,警方通過查閱死者的電腦和手機篡腌,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,581評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來勾效,“玉大人嘹悼,你說我怎么就攤上這事〔愎” “怎么了杨伙?”我有些...
    開封第一講書人閱讀 165,830評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長萌腿。 經(jīng)常有香客問我限匣,道長,這世上最難降的妖魔是什么毁菱? 我笑而不...
    開封第一講書人閱讀 58,957評論 1 295
  • 正文 為了忘掉前任米死,我火速辦了婚禮,結(jié)果婚禮上贮庞,老公的妹妹穿的比我還像新娘峦筒。我一直安慰自己,他們只是感情好贸伐,可當(dāng)我...
    茶點故事閱讀 67,974評論 6 393
  • 文/花漫 我一把揭開白布勘天。 她就那樣靜靜地躺著怔揩,像睡著了一般捉邢。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上商膊,一...
    開封第一講書人閱讀 51,754評論 1 307
  • 那天伏伐,我揣著相機與錄音,去河邊找鬼晕拆。 笑死藐翎,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的实幕。 我是一名探鬼主播吝镣,決...
    沈念sama閱讀 40,464評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼昆庇!你這毒婦竟也來了末贾?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,357評論 0 276
  • 序言:老撾萬榮一對情侶失蹤整吆,失蹤者是張志新(化名)和其女友劉穎拱撵,沒想到半個月后辉川,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,847評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡拴测,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,995評論 3 338
  • 正文 我和宋清朗相戀三年乓旗,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片集索。...
    茶點故事閱讀 40,137評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡屿愚,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出务荆,到底是詐尸還是另有隱情渺鹦,我是刑警寧澤,帶...
    沈念sama閱讀 35,819評論 5 346
  • 正文 年R本政府宣布蛹含,位于F島的核電站毅厚,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏浦箱。R本人自食惡果不足惜吸耿,卻給世界環(huán)境...
    茶點故事閱讀 41,482評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望酷窥。 院中可真熱鬧咽安,春花似錦、人聲如沸蓬推。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,023評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽沸伏。三九已至糕珊,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間毅糟,已是汗流浹背红选。 一陣腳步聲響...
    開封第一講書人閱讀 33,149評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留姆另,地道東北人喇肋。 一個月前我還...
    沈念sama閱讀 48,409評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像迹辐,于是被迫代替她去往敵國和親蝶防。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,086評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 從三月份找實習(xí)到現(xiàn)在明吩,面了一些公司间学,掛了不少,但最終還是拿到小米、百度菱鸥、阿里宗兼、京東、新浪氮采、CVTE殷绍、樂視家的研發(fā)崗...
    時芥藍閱讀 42,255評論 11 349
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法鹊漠,內(nèi)部類的語法主到,繼承相關(guān)的語法,異常的語法躯概,線程的語...
    子非魚_t_閱讀 31,644評論 18 399
  • java筆記第一天 == 和 equals ==比較的比較的是兩個變量的值是否相等登钥,對于引用型變量表示的是兩個變量...
    jmychou閱讀 1,502評論 0 3
  • 一牧牢、基本數(shù)據(jù)類型 注釋 單行注釋:// 區(qū)域注釋:/* */ 文檔注釋:/** */ 數(shù)值 對于byte類型而言...
    龍貓小爺閱讀 4,265評論 0 16
  • 好像每次都是這樣,沒有例外姿锭。在我們最需要有一個人去依靠的時候塔鳍,往往到最后都是自己一個人挺過去。 ??? 這是為什么...
    A分享閱讀 490評論 1 1