似懂非懂的Comparable與Comparator

来源:https://www.cnblogs.com/yulinfeng/archive/2018/02/22/8460569.html
-Advertisement-
Play Games

jdk1.8.0_41 一知半解寫代碼, 集合排序用個啥。 抄起鍵盤胡亂打, 似懂非懂最可怕。 Comparable與Comparator都是用於集合的排序,對於大多數人來說Comparator可能略微比Comparable要熟悉一點,類似下麵這幾句代碼的使用頻率應該是最高的。 這是一段對集合排序的 ...


jdk1.8.0_144

一知半解寫代碼, 集合排序用個啥。 抄起鍵盤胡亂打, 似懂非懂最可怕。

  Comparable與Comparator都是用於集合的排序,對於大多數人來說Comparator可能略微比Comparable要熟悉一點,類似下麵這幾句代碼的使用頻率應該是最高的。

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

  這是一段對集合排序的代碼。

  對於集合排序時比較器的使用往往止步於此,以至於更為深層次的使用似懂非懂,更為複雜的排序不知所措。

  Comparable用於集合內部定義的方法實現的排序,Comparator用於集合外部實現的排序。

  我們從Collections.sort的兩個重載方法開始

public static <T extends Comparable<? super T>> void sort(List<T> list)

public static <T> void sort(List<T> list, Comparator<? super T> c)

  這兩個方法都是泛型方法,第一個方法只傳遞一個List參數進行排序,第二個方法傳遞一個List參數加上一個Comparator比較器。

 

public static <T extends Comparable<? super T>> void sort(List<T> list)

  Collections.sort方法用於對List集合進行排序,思考一個問題,這個只有一個參數的List如何進行排序呢?它是按照怎樣的規則進行排序呢?答案就在這個泛型方法的泛型類型之中“<T extends Comparable<? super T>>”,List集合中的元素需要實現Comparable介面,Comparable介面也是一個泛型,並要求它的泛型類型需要是集合中的元素的超類(或自身)。重點在於——集合中的元素需要是實現Comparable介面。也就是說在使用Collections.sort(List)這個方法對集合中的元素進行排序時,需要集合中的元素實現了Comparable介面,這才能進行排序。

 

public static <T> void sort(List<T> list, Comparator<? super T> c)

  這個方法同樣是一個泛型方法,與上面的方法不同的是對集合中的元素類型並沒有做限制,要對這個集合進行排序需要指定一個Comparator比較器,這個比較器的泛型類型需要是集合元素的超類(或自身)。

  通過上面這兩個方法比較容易的能得出一個淺顯的結論,Comparable和Comparator都是用於比較、排序,如果元素自身已經實現了Comparable介面,則可以利用它自身進行比較排序,如果元素自身沒有實現Comparable介面,則可以利用外部實現Comparator比較器對元素進行比較排序。這也就是前面提到的Comparable用於集合內部定義的方法實現的排序,Comparator用於集合外部實現的排序。 接著來看Comparable介面和Comparator介面。

 

  接著來看Comparable介面和Comparator介面。

Comparable

public int compareTo(T o)

  這個介面只定義了一個compareTo方法,在很多“值類型”的數據類型,例如String、Integer、Long等已經實現了這個介面。

  對於這個方法和equals方法有類似的地方,equals強調的更多是相等於否,而compareTo強調更多的比較,如果x < y,則返回-1;x = y,則返回0;x > y,則返回1。實現這個方法時同樣需要遵循幾個規則:

  1. 自反性,如果x <y ,那麼x.compareTo(y)返回-1,同理y.compareTo則返回1;
  2. 傳遞性,如果x.compareTo(y)返回-1,y.compareTo(z)返回-1,則x.compareTo(z)也應該返回-1;
  3. 同一性,如果x.compareTo(y)返回0(x = y),那麼若x.compareTo(z)返回-1時,y.compareTo(z)也應該返回-1。

  有興趣的可以查看String類中對於compareTo方法的實現,它的排序規則是將字元串轉換為字元數組逐個按照字典序排序。

 

Comparator

int compare(T o1, T o2)

boolean equals(Object obj)

  這個介面在JDK8中對它進行了較大的改進,在JDK8之前只包含上面兩個方法,而JDK8則達到了18個方法,其中都是介面的default方法,和static靜態方法,所以並不需要在實現時額外實現。

  在JDK8中該類添加了@FunctionalInterface函數式介面的註解,函數式介面表明在介面中只含有一個方法作為Lambda表達式的數據類型,在《JDK8的新特性——Lambda表達式》中有提到,Java中定義如果覆蓋了Object中的方法則不算,所以在Comparator介面中只有一個compare方法。對於@FunctionalInteface註解可加不可加,加上只是為了讓編譯器做更好的檢查,要求只能定義一個方法,不加編譯器便不對此進行檢查。

  compare方法和compareTo方法類似,它同樣需要滿足上面提到的:自反性、傳遞性、同一性。並且它強調,不必嚴格滿足“(compare(x, y)==0) == (x.equals(y))”,當然最好說明白。

  對於這個類,更多的是需要理解學習它所運用的設計模式——策略模式。策略模式,不改變對象自身,而是用另一個對象來改變它的行為。例如,超市減價操作,有10件商品需要統一降價1半進行處理,我們可以在這10件商品的價格上全部做修改減少至它的一半,10件好處理,如果N件呢,甚至我們還需要對其進行降價呢?此時我們則可以使用一種“策略”——商品全部打5折。這就是利用另外一個對象來改變一個對象的行為,而不是簡單粗暴地修改原有對象。說回此處,如果List中的元素本身沒有實現Comparable介面,但我們需要對它進行排序,我們可以對原有對象進行修改讓它實現Comparable介面,但凡涉及修改代碼都不優美,此時我們則可以利用策略模式,也就是實現一個Comparator介面對集合中的元素進行排序。

 

 

這是一個能給程式員加buff的公眾號 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 瀑布流佈局是一個之前挺火,現在也比較常見的一種佈局 這種看起來參差不齊的多欄佈局,重點在於每個元素等寬不等高,一般圖片網站應用較多 隨著頁面滾動,數據塊會在每列底部不斷載入,理論上可以無限載入,且不會頁面美觀 Pinterest、花瓣網都是比較成熟的採用瀑布流佈局的網站 效果圖: PS:這篇博客只有 ...
  • 1.BFC 定義 BFC(Block formatting context)直譯為"塊級格式化上下文"。它是一個獨立的渲染區域,只有Block level box(塊級元素)參與, 它規定了內部的Block level Box如何佈局,並且與這個區域外部毫不相干. 通俗地來說:創建了 BFC的元素就 ...
  • 一.前言 目前的轉轉app是一個典型的hybrid app,採用的是業內主流的做法: 客戶端內有大量業務頁面使用webview內載入h5頁面承載。 其優點是顯而易見的,即:web頁面上線頻度滿足快速迭代的業務需求,不受客戶端審核和發版的時間限制,也可以將各個業務線的開發工作分攤到各個業務的fe團隊上 ...
  • 第一次嘗試使用Office Word,方便程度大大超過網頁線上編輯,不過初次使用,一些內容不甚熟悉,望各位大神見諒~ 在上次的文章中,把整個jQuery的結構進行了梳理,得到了整個jQuery的簡化結構,如下: 這一次,我們將開始對jQuery源碼進行逐行分解。 首先從一開始的註釋開始 開始的註釋包 ...
  • 在之前寫過的一篇《"天龍八步"細說瀏覽器輸入URL後發生了什麼》一文中,和大家分享了從在瀏覽器中輸入網址URL到最終頁面展示的整個過程。部分讀者向我反饋對於最後的瀏覽器渲染佈局這塊不是很清晰,所以本文就瀏覽器渲染流程單獨開篇講解,希望大家都能有新的收穫。 瀏覽器主要組件結構 (瀏覽器主要組件) 渲染 ...
  • 標準w3c盒模型圖示 關於怪異盒模型和標準盒模型的對比 關於怪異盒模型盒標準盒模型之間的轉換 ...
  • 數組排序演算法 (一)冒泡排序 基本思想:對比相鄰的元素值,如果滿足條件就交換元素值,把較小的元素移動數組前面,較大的元素移動到數組後面演算法:冒泡演算法由雙層迴圈實現,其中外層迴圈控制排序輪次,一般為排序的數組長度減一。而內層迴圈主要用於對比數組中每個臨近元素的大小,以確定是否交換位置,對比和交換的次數 ...
  • (一) 前言 突然的資源受限或網路延遲,可能導致找不到目標元素,這時測試報告會顯示測試失敗。這時需要一種延時機制,來使腳本的運行速度與程式的響應速度相匹配,WebDriver為這種情況提供了隱式等待和顯式等待兩種機制。 (二) 隱式等待 一旦設置隱式等待時間,就會作用於這個WebDriver實例的整 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...