欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java 中Comparable與Comparator詳解與比較

 更新時(shí)間:2017年04月08日 14:23:40   投稿:lqh  
這篇文章主要介紹了java 中Comparable與Comparator詳解與比較的相關(guān)資料,需要的朋友可以參考下

java 中Comparable與Comparator詳解

今天查看TreeMap的源碼,發(fā)現(xiàn)其鍵必須是實(shí)現(xiàn)Comparable或者Comparator的接口時(shí)產(chǎn)生了一些興趣,比如在TreeMap中的put方法分別對Comparable和Comparator接口分別進(jìn)行處理。那么疑問就來了,Comparable和Comparator接口的區(qū)別是什么,Java中為什么會(huì)存在兩個(gè)類似的接口?

  Comparable和Comparator接口都是用來比較大小的,首先來看一下Comparable的定義:

 package java.lang;
import java.util.*;
public interface Comparable<T> {
  public int compareTo(T o);
}

  Comparator的定義如下:

package java.util;
public interface Comparator<T> {
  int compare(T o1, T o2);
  boolean equals(Object obj);
}

  Comparable對實(shí)現(xiàn)它的每個(gè)類的對象進(jìn)行整體排序。這個(gè)接口需要類本身去實(shí)現(xiàn)(這句話沒看懂?沒關(guān)系,接下來看個(gè)例子就明白了)。若一個(gè)類實(shí)現(xiàn)了Comparable 接口,實(shí)現(xiàn) Comparable 接口的類的對象的 List 列表 ( 或數(shù)組)可以通過 Collections.sort(或 Arrays.sort)進(jìn)行排序。此外,實(shí)現(xiàn) Comparable 接口的類的對象 可以用作 “有序映射 ( 如 TreeMap)” 中的鍵或 “有序集合 (TreeSet)” 中的元素,而不需要指定比較器。

  舉例(類Person1實(shí)現(xiàn)了Comparable接口)

package collections;

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

  public Person1(String name, int age)
  {
    this.name = name;
    this.age = age;
  }
  @Override
  public int compareTo(Person1 o)
  {
    return this.age-o.age;
  }
  @Override 
  public String toString()
  {
    return name+":"+age;
  }
}

  可以看到Person1實(shí)現(xiàn)了Comparable接口中的compareTo方法。實(shí)現(xiàn)Comparable接口必須修改自身的類,即在自身類中實(shí)現(xiàn)接口中相應(yīng)的方法。

  測試代碼:

 Person1 person1 = new Person1("zzh",18);
    Person1 person2 = new Person1("jj",17);
    Person1 person3 = new Person1("qq",19);

    List<Person1> list = new ArrayList<>();
    list.add(person1);
    list.add(person2);
    list.add(person3);

    System.out.println(list);
    Collections.sort(list);
    System.out.println(list);

  輸出結(jié)果:

[zzh:18, jj:17, qq:19]
[jj:17, zzh:18, qq:19]

  如果我們的這個(gè)類無法修改,譬如String,我們又要對其進(jìn)行排序,當(dāng)然String中已經(jīng)實(shí)現(xiàn)了Comparable接口,如果單純的用String舉例就不太形象。對類自身無法修改這就用到了Comparator這個(gè)接口(策略模式)。

public final class Person2
{
  private int age;
  private String name;

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

  @Override 
  public String toString()
  {
    return name+":"+age;
  }

  //getter and setter方法省略....
}

  如類Person2,這個(gè)類已經(jīng)固定,無法進(jìn)行對其類自身的修改,也修飾詞final了,你也別想繼承再implements Comparable,那么此時(shí)怎么辦呢?在類的外部使用Comparator的接口。如下測試代碼:

 Person2 p1 = new Person2("zzh",18);
    Person2 p2 = new Person2("jj",17);
    Person2 p3 = new Person2("qq",19);
    List<Person2> list2 = new ArrayList<Person2>();
    list2.add(p1);
    list2.add(p2);
    list2.add(p3);
    System.out.println(list2);
    Collections.sort(list2,new Comparator<Person2>(){

      @Override
      public int compare(Person2 o1, Person2 o2)
      {
        if(o1 == null || o2 == null)
          return 0;
        return o1.getAge()-o2.getAge();
      }

    });
    System.out.println(list2);

  輸出結(jié)果:

[zzh:18, jj:17, qq:19]
[jj:17, zzh:18, qq:19] 

  這里(public static <T> void sort(List<T> list, Comparator<? super T> c) )采用了內(nèi)部類的實(shí)現(xiàn)方式,實(shí)現(xiàn)compare方法,對類Person2的list進(jìn)行排序。

  再譬如博主遇到的真實(shí)案例中,需要對String進(jìn)行排序,且不區(qū)分大小寫,我們知道String中的排序是字典排序,譬如:A a D排序之后為A D a,這樣顯然不對,那么該怎么辦呢?同上(下面代碼中的list是一個(gè)String的List集合):

 Collections.sort(list, new Comparator<String>()
    {
      @Override
      public int compare(String o1, String o2)
      {
        if(o1 == null || o2 == null)
          return 0;
        return o1.toUpperCase().compareTo(o2.toUpperCase());
      }
    });

  這樣就可以實(shí)現(xiàn)不區(qū)分大小進(jìn)行排序String的集合了,是不是很方便~

  細(xì)心的同學(xué)可能會(huì)有疑問,明明在Comparator接口中定義了兩個(gè)方法,為什么繼承的時(shí)候只實(shí)現(xiàn)了一個(gè)方法,難道要顛覆我對Java接口常識的理解了???

  實(shí)際上,我們知道當(dāng)一個(gè)類沒有顯式繼承父類的時(shí)候,會(huì)有一個(gè)默認(rèn)的父類,即java.lang.Object,在Object類中有一個(gè)方法即為equals方法,所以這里并不強(qiáng)制要求實(shí)現(xiàn)Comparator接口的類要實(shí)現(xiàn)equals方法,直接調(diào)用父類的即可,雖然你顯式的實(shí)現(xiàn)了equals()方法 will be a better choice~

  在《Effective Java》一書中,作者Joshua Bloch推薦大家在編寫自定義類的時(shí)候盡可能的考慮實(shí)現(xiàn)一下Comparable接口,一旦實(shí)現(xiàn)了Comparable接口,它就可以跟許多泛型算法以及依賴于改接口的集合實(shí)現(xiàn)進(jìn)行協(xié)作。你付出很小的努力就可以獲得非常強(qiáng)大的功能。

  事實(shí)上,Java平臺類庫中的所有值類都實(shí)現(xiàn)了Comparable接口。如果你正在編寫一個(gè)值類,它具有非常明顯的內(nèi)在排序關(guān)系,比如按字母順序、按數(shù)值順序或者按年代順序,那你就應(yīng)該堅(jiān)決考慮實(shí)現(xiàn)這個(gè)接口。

  compareTo方法不但允許進(jìn)行簡單的等同性進(jìn)行比較,而且語序執(zhí)行順序比較,除此之外,它與Object的equals方法具有相似的特征,它還是一個(gè)泛型。類實(shí)現(xiàn)了Comparable接口,就表明它的實(shí)例具有內(nèi)在的排序關(guān)系,為實(shí)現(xiàn)Comparable接口的對象數(shù)組進(jìn)行排序就這么簡單: Arrays.sort(a);

  對存儲在集合中的Comparable對象進(jìn)行搜索、計(jì)算極限值以及自動(dòng)維護(hù)也同樣簡單。列如,下面的程序依賴于String實(shí)現(xiàn)了Comparable接口,它去掉了命令行參數(shù)列表中的重復(fù)參數(shù),并按字母順序打印出來:

public class WordList{
  public static void main(String args[]){
    Set<String> s = new TreeSet<String>();
    Collections.addAll(s,args);
    System.out.println(s);
  }
}

  Comparable 是排序接口;若一個(gè)類實(shí)現(xiàn)了 Comparable 接口,就意味著 “該類支持排序”。而 Comparator 是比較器;我們?nèi)粜枰刂颇硞€(gè)類的次序,可以建立一個(gè) “該類的比較器” 來進(jìn)行排序。

  前者應(yīng)該比較固定,和一個(gè)具體類相綁定,而后者比較靈活,它可以被用于各個(gè)需要比較功能的類使用??梢哉f前者屬于 “靜態(tài)綁定”,而后者可以 “動(dòng)態(tài)綁定”。

  我們不難發(fā)現(xiàn):Comparable 相當(dāng)于 “內(nèi)部比較器”,而 Comparator 相當(dāng)于 “外部比較器”。

感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!

相關(guān)文章

  • maven打包本地jar到項(xiàng)目中的方法實(shí)現(xiàn)

    maven打包本地jar到項(xiàng)目中的方法實(shí)現(xiàn)

    本文主要介紹了maven打包本地jar到項(xiàng)目中的方法實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • 使用BigDecimal進(jìn)行精確運(yùn)算(實(shí)現(xiàn)加減乘除運(yùn)算)

    使用BigDecimal進(jìn)行精確運(yùn)算(實(shí)現(xiàn)加減乘除運(yùn)算)

    這篇文章主要介紹了如何使用BigDecimal進(jìn)行精確運(yùn)算,最后提供了一個(gè)工具類,該工具類提供加,減,乘,除運(yùn)算
    2013-11-11
  • java 實(shí)例化類詳解及簡單實(shí)例

    java 實(shí)例化類詳解及簡單實(shí)例

    這篇文章主要介紹了java 實(shí)例化類詳解及簡單實(shí)例的相關(guān)資料,需要的朋友可以參考下
    2017-03-03
  • Java解析pdf格式發(fā)票的代碼實(shí)現(xiàn)

    Java解析pdf格式發(fā)票的代碼實(shí)現(xiàn)

    為了減少用戶工作量及誤操作的可能性,需要實(shí)現(xiàn)用戶上傳PDF格式的發(fā)票,系統(tǒng)通過解析PDF文件獲取發(fā)票內(nèi)容,并直接將其寫入表單,以下文章記錄了功能實(shí)現(xiàn)的代碼,需要的朋友可以參考下
    2024-08-08
  • Spring Security認(rèn)證機(jī)制源碼層探究

    Spring Security認(rèn)證機(jī)制源碼層探究

    SpringSecurity是基于Filter實(shí)現(xiàn)認(rèn)證和授權(quán),底層通過FilterChainProxy代理去調(diào)用各種Filter(Filter鏈),F(xiàn)ilter通過調(diào)用AuthenticationManager完成認(rèn)證 ,通過調(diào)用AccessDecisionManager完成授權(quán)
    2023-03-03
  • Mybatis日志配置方式(slf4j、log4j、log4j2)

    Mybatis日志配置方式(slf4j、log4j、log4j2)

    這篇文章主要介紹了Mybatis日志配置方式(slf4j、log4j、log4j2),具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • 5種Java中數(shù)組的拷貝方法總結(jié)分享

    5種Java中數(shù)組的拷貝方法總結(jié)分享

    這篇文章主要介紹了5種Java中數(shù)組的拷貝方法總結(jié)分享,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的朋友可以參考一下
    2022-07-07
  • Springboot如何使用Aspectj實(shí)現(xiàn)AOP面向切面編程

    Springboot如何使用Aspectj實(shí)現(xiàn)AOP面向切面編程

    這篇文章主要介紹了Springboot如何使用Aspectj實(shí)現(xiàn)AOP面向切面編程,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-01-01
  • 關(guān)于spring中定時(shí)器的使用教程

    關(guān)于spring中定時(shí)器的使用教程

    大家應(yīng)該都有所體會(huì),在很多實(shí)際的web應(yīng)用中,都有需要定時(shí)實(shí)現(xiàn)的服務(wù),下面這篇文章主要給大家介紹了關(guān)于spring中定時(shí)器的使用教程,對大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起看看吧。
    2017-06-06
  • JAVA實(shí)現(xiàn)生成短鏈接的示例代碼

    JAVA實(shí)現(xiàn)生成短鏈接的示例代碼

    短鏈接就是將長度較長的鏈接壓縮成較短的鏈接,本文就來介紹一下JAVA實(shí)現(xiàn)生成短鏈接的示例代碼,具有一定的參考價(jià)值,感興趣的可以了解一下
    2023-08-08

最新評論