国产av日韩一区二区三区精品,成人性爱视频在线观看,国产,欧美,日韩,一区,www.成色av久久成人,2222eeee成人天堂

首頁 Java java教程 Java 中的比較器是如何工作的?

Java 中的比較器是如何工作的?

Nov 05, 2024 pm 04:52 PM

?Cómo funciona Comparator en Java?

介紹

有時在做項目時需要對某種類型的對象集合進行排序,為此你可能會認為有必要實現(xiàn)我們自己的排序算法,但這有點不必要,盡管了解一下也沒什么壞處他們如何工作。例如,如果您有一個整數(shù)數(shù)組,則可以使用 Arrays.sort() 方法,該方法接受基元數(shù)組并按升序對其進行排序,從而利用無需將結果分配給新變量,因為該方法修改了原始數(shù)組。

int[] numbers = {9, 8, 5, 3, 1, 2, 4, 6, 7};
Arrays.sort(numbers);
System.out.println(Arrays.toString(numbers));

// Output
[1, 2, 3, 4, 5, 6, 7, 8, 9]

當您有自定義對象的集合(例如 Movie 類型的記錄)時,這也適用,但如果我們看到 Arrays.sort() 方法,它不接受這種類型的對象數(shù)組,因此它必須使用 sort() 方法接受 T 類型的對象和 Comparator 類型的對象作為參數(shù)超級T>這是一個函數(shù)式接口。這個接口非常重要,因為 Java 中的許多其他方法都使用它以自定義方式比較對象。例如,List 對象的 Collections.sort() 方法或 sort() 方法,甚至 Streams 也接受 Comparator 來對元素進行排序。

什么是比較器?

函數(shù)式接口Comparator(函數(shù)式它可以寫成lambda表達式)是一個允許你比較兩個T類型對象的接口,因此它用于比較整數(shù)、字符串、自定義物體等該接口有幾個靜態(tài)和默認方法,但重要的是compare() 方法,它是比較兩個對象必須實現(xiàn)的方法。 Compare() 接收兩個 T 類型的對象并返回一個整數(shù)。方法簽名如下:

int compare(T o1, T o2);

如果o1小于o2,該方法返回負數(shù);如果相等,則返回零;如果o1大于o2,該方法返回正數(shù),通常分別返回-1、0或1。

一個對象小于、等于或大于另一個對象意味著什么?

讓我們分析一下compare()方法返回的內容,因為對象的排序取決于此,重要的是要考慮該方法返回的含義是相對的,也就是說,如果您想要升序或降序排列。這取決于具體情況以及如何實施。讓我們?yōu)槊總€示例考慮以下記錄

int[] numbers = {9, 8, 5, 3, 1, 2, 4, 6, 7};
Arrays.sort(numbers);
System.out.println(Arrays.toString(numbers));

// Output
[1, 2, 3, 4, 5, 6, 7, 8, 9]
  • 如果第一個參數(shù)小于第二個參數(shù),則返回負數(shù)。例如,要按發(fā)行年份對電影進行排序,當電影 a 小于電影 b 時,可以返回 -1:
int compare(T o1, T o2);
  • 如果第一個參數(shù)大于第二個參數(shù),則返回正數(shù)。例如,要按預算對電影進行排序,當電影 a 大于電影 b 時,可以返回 1:
public record Movie(
        String name,
        List<String> actors,
        int budget,
        int year
) {
}
  • 如果第一個參數(shù)等于第二個參數(shù),則返回零。例如,要按演員數(shù)量對電影進行排序,當電影 a 等于電影 b 時,可以返回 0:
// a < b -> -1
a.year() < b.year() -> -1

使用比較器

假設我們在 List 類型的對象中有以下電影:

// a > b -> 1
a.budget() > b.budget() -> 1

如果您想按發(fā)行年份升序排列電影,您可以創(chuàng)建一個 Comparator 類型的對象并重寫compare()方法,然后將此對象傳遞給列表的sort()方法:

// a == b -> 0
a.actors().size() == b.actors().size() -> 0

它也可以在 sort() 方法中實現(xiàn)為匿名類:

Movie movie1 = new Movie("The Godfather", Arrays.asList("Marlon Brando", "Al Pacino"), 6000000, 1972);
Movie movie2 = new Movie("The Godfather: Part II", Arrays.asList("Al Pacino", "Robert De Niro"), 13000000, 1974);
Movie movie3 = new Movie("The Shawshank Redemption", Arrays.asList("Tim Robbins", "Morgan Freeman"), 25000000, 1994);
Movie movie4 = new Movie("The Dark Knight", Arrays.asList("Christian Bale", "Heath Ledger"), 185000000, 2008);

List<Movie> movies = Arrays.asList(movie1, movie2, movie3, movie4);

或者更簡潔地直接在 sort() 方法中使用 lambda 表達式:

Comparator<Movie> comparatorByYear = new Comparator<Movie>() {
    @Override
    public int compare(Movie o1, Movie o2) {
        return o1.year() - o2.year();
    }
};

movies.sort(comparatorByYear);

任何這些實現(xiàn)??都會按發(fā)布年份對列表進行升序排序。如果要按降序排序,可以更改 lambda 表達式中參數(shù)的順序,或者在減法中添加負號:

movies.sort(new Comparator<Movie>() {
    @Override
    public int compare(Movie o1, Movie o2) {
        return o1.year() - o2.year();
    }
});

如何對自定義對象列表進行排序的一些其他示例是:

  • 按演員數(shù)量升序排列電影(從少到多):
movies.sort((p1, p2) -> p1.year() - p2.year());
  • 按預算降序對電影進行排序(從最高預算到最低預算):
movies.sort((p1, p2) -> p2.year() - p1.year());
// o
movies.sort((p1, p2) -> - (p1.year() - p2.year()));
  • 按名稱升序對電影進行排序:
movies.sort((p1, p2) -> p1.actors().size() - p2.actors().size());

在其他示例中,我們可能會遇到需要按降序對整數(shù)列表進行排序的情況,

movies.sort((p1, p2) -> p2.budget() - p1.budget());
// o 
movies.sort((p1, p2) -> - (p1.budget() - p2.budget()));

為此,您還可以使用靜態(tài)方法 Comparator.reverseOrder() 返回一個按降序對元素進行排序的比較器,以及使用 Comparator.naturalOrder() 按升序對元素進行排序。

int[] numbers = {9, 8, 5, 3, 1, 2, 4, 6, 7};
Arrays.sort(numbers);
System.out.println(Arrays.toString(numbers));

// Output
[1, 2, 3, 4, 5, 6, 7, 8, 9]

使用 Integer.compare()

在 Java 中已經(jīng)有一些方法允許我們以有效的方式執(zhí)行這種類型的比較,例如 Integer.compare() ,它比較兩個整數(shù),如果第一個參數(shù)小于第二個參數(shù)零,則返回負數(shù)如果第一個參數(shù)大于第二個參數(shù),則相等且為正數(shù)。如果我們分析這個方法的工作原理,我們可以看到它與上面解釋的類似,并且返回的正是Comparator接口的compare()方法所需要的。 Integer.compare() 的實現(xiàn)如下:

int compare(T o1, T o2);

因此,如果您想按發(fā)行年份升序對電影進行排序,可以使用 Integer.compare():

public record Movie(
        String name,
        List<String> actors,
        int budget,
        int year
) {
}

使用參考方法

有時可以使用引用方法來執(zhí)行與以前不同的比較,例如,按升序對整數(shù)列表進行排序:

// a < b -> -1
a.year() < b.year() -> -1

Integer 不是唯一具有compareTo() 方法的類,例如 String 有一個compareTo() 方法,可以按字典順序比較兩個字符串,因此它可以用于對字符串列表進行排序,甚至可以將 CharSequence 與其一起使用Compare() 方法(技術上表示字符序列)。

// a > b -> 1
a.budget() > b.budget() -> 1

回到電影的例子,如果你想按照上映年份升序對電影進行排序,可以使用compareingInt()方法作為參考方法:

// a == b -> 0
a.actors().size() == b.actors().size() -> 0

或者根據(jù) String 類型屬性進行比較,在本例中為電影名稱:

Movie movie1 = new Movie("The Godfather", Arrays.asList("Marlon Brando", "Al Pacino"), 6000000, 1972);
Movie movie2 = new Movie("The Godfather: Part II", Arrays.asList("Al Pacino", "Robert De Niro"), 13000000, 1974);
Movie movie3 = new Movie("The Shawshank Redemption", Arrays.asList("Tim Robbins", "Morgan Freeman"), 25000000, 1994);
Movie movie4 = new Movie("The Dark Knight", Arrays.asList("Christian Bale", "Heath Ledger"), 185000000, 2008);

List<Movie> movies = Arrays.asList(movie1, movie2, movie3, movie4);

按多個屬性排序

有時你可能需要按多個屬性對對象列表進行排序,例如,如果你想按發(fā)行年份升序對電影進行排序,按預算降序排序,你可以使用 thenComparing() 方法,該方法接收一個比較器,負責按多個屬性排序。例如,如果有兩部電影a和b,上映年份相同,則按預算排序。

Comparator<Movie> comparatorByYear = new Comparator<Movie>() {
    @Override
    public int compare(Movie o1, Movie o2) {
        return o1.year() - o2.year();
    }
};

movies.sort(comparatorByYear);

結論

比較器在Java中很多時候都很有用,因為它們可以讓你以個性化的方式比較對象,不僅如此,它們還可以用在許多Java集合方法中,甚至可以有多個比較器來排序以不同的方式。無論哪種方式,您都可以查閱 IDE 中的 Comparator 文檔或官方 Java 文檔,了解可以使用哪些方法以及如何實現(xiàn)它們。

以上是Java 中的比較器是如何工作的?的詳細內容。更多信息請關注PHP中文網(wǎng)其他相關文章!

本站聲明
本文內容由網(wǎng)友自發(fā)貢獻,版權歸原作者所有,本站不承擔相應法律責任。如您發(fā)現(xiàn)有涉嫌抄襲侵權的內容,請聯(lián)系admin@php.cn

熱AI工具

Undress AI Tool

Undress AI Tool

免費脫衣服圖片

Undresser.AI Undress

Undresser.AI Undress

人工智能驅動的應用程序,用于創(chuàng)建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用于從照片中去除衣服的在線人工智能工具。

Clothoff.io

Clothoff.io

AI脫衣機

Video Face Swap

Video Face Swap

使用我們完全免費的人工智能換臉工具輕松在任何視頻中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的代碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

功能強大的PHP集成開發(fā)環(huán)境

Dreamweaver CS6

Dreamweaver CS6

視覺化網(wǎng)頁開發(fā)工具

SublimeText3 Mac版

SublimeText3 Mac版

神級代碼編輯軟件(SublimeText3)

hashmap和hashtable之間的區(qū)別? hashmap和hashtable之間的區(qū)別? Jun 24, 2025 pm 09:41 PM

HashMap與Hashtable的區(qū)別主要體現(xiàn)在線程安全、null值支持及性能方面。1.線程安全方面,Hashtable是線程安全的,其方法大多為同步方法,而HashMap不做同步處理,非線程安全;2.null值支持上,HashMap允許一個null鍵和多個null值,Hashtable則不允許null鍵或值,否則拋出NullPointerException;3.性能方面,HashMap因無同步機制效率更高,Hashtable因每次操作加鎖性能較低,推薦使用ConcurrentHashMap替

為什么我們需要包裝紙課? 為什么我們需要包裝紙課? Jun 28, 2025 am 01:01 AM

Java使用包裝類是因為基本數(shù)據(jù)類型無法直接參與面向對象操作,而實際需求中常需對象形式;1.集合類只能存儲對象,如List利用自動裝箱存儲數(shù)值;2.泛型不支持基本類型,必須使用包裝類作為類型參數(shù);3.包裝類可表示null值,用于區(qū)分未設置或缺失的數(shù)據(jù);4.包裝類提供字符串轉換等實用方法,便于數(shù)據(jù)解析與處理,因此在需要這些特性的場景下,包裝類不可或缺。

什么是接口中的靜態(tài)方法? 什么是接口中的靜態(tài)方法? Jun 24, 2025 pm 10:57 PM

StaticmethodsininterfaceswereintroducedinJava8toallowutilityfunctionswithintheinterfaceitself.BeforeJava8,suchfunctionsrequiredseparatehelperclasses,leadingtodisorganizedcode.Now,staticmethodsprovidethreekeybenefits:1)theyenableutilitymethodsdirectly

JIT編譯器如何優(yōu)化代碼? JIT編譯器如何優(yōu)化代碼? Jun 24, 2025 pm 10:45 PM

JIT編譯器通過方法內聯(lián)、熱點檢測與編譯、類型推測與去虛擬化、冗余操作消除四種方式優(yōu)化代碼。1.方法內聯(lián)減少調用開銷,將頻繁調用的小方法直接插入調用處;2.熱點檢測識別高頻執(zhí)行代碼并集中優(yōu)化,節(jié)省資源;3.類型推測收集運行時類型信息實現(xiàn)去虛擬化調用,提升效率;4.冗余操作消除根據(jù)運行數(shù)據(jù)刪除無用計算和檢查,增強性能。

什么是實例初始器塊? 什么是實例初始器塊? Jun 25, 2025 pm 12:21 PM

實例初始化塊在Java中用于在創(chuàng)建對象時運行初始化邏輯,其執(zhí)行先于構造函數(shù)。它適用于多個構造函數(shù)共享初始化代碼、復雜字段初始化或匿名類初始化場景,與靜態(tài)初始化塊不同的是它每次實例化時都會執(zhí)行,而靜態(tài)初始化塊僅在類加載時運行一次。

變量的最終關鍵字是什么? 變量的最終關鍵字是什么? Jun 24, 2025 pm 07:29 PM

InJava,thefinalkeywordpreventsavariable’svaluefrombeingchangedafterassignment,butitsbehaviordiffersforprimitivesandobjectreferences.Forprimitivevariables,finalmakesthevalueconstant,asinfinalintMAX_SPEED=100;wherereassignmentcausesanerror.Forobjectref

什么是工廠模式? 什么是工廠模式? Jun 24, 2025 pm 11:29 PM

工廠模式用于封裝對象創(chuàng)建邏輯,使代碼更靈活、易維護、松耦合。其核心答案是:通過集中管理對象創(chuàng)建邏輯,隱藏實現(xiàn)細節(jié),支持多種相關對象的創(chuàng)建。具體描述如下:工廠模式將對象創(chuàng)建交給專門的工廠類或方法處理,避免直接使用newClass();適用于多類型相關對象創(chuàng)建、創(chuàng)建邏輯可能變化、需隱藏實現(xiàn)細節(jié)的場景;例如支付處理器中通過工廠統(tǒng)一創(chuàng)建Stripe、PayPal等實例;其實現(xiàn)包括工廠類根據(jù)輸入?yún)?shù)決定返回的對象,所有對象實現(xiàn)共同接口;常見變體有簡單工廠、工廠方法和抽象工廠,分別適用于不同復雜度的需求。

什么是類型鑄造? 什么是類型鑄造? Jun 24, 2025 pm 11:09 PM

類型轉換有兩種:隱式和顯式。1.隱式轉換自動發(fā)生,如將int轉為double;2.顯式轉換需手動操作,如使用(int)myDouble。需要類型轉換的情況包括處理用戶輸入、數(shù)學運算或函數(shù)間傳遞不同類型的值時。需要注意的問題有:浮點數(shù)轉整數(shù)會截斷小數(shù)部分、大類型轉小類型可能導致數(shù)據(jù)丟失、某些語言不允許直接轉換特定類型。正確理解語言的轉換規(guī)則有助于避免錯誤。

See all articles