【Java必修課】一圖說盡排序,一文細說Sorting(Array、List、Stream的排序)

簡說排序

排序是極其常見的使用場景,因為在生活中就有很多這樣的實例。國家GDP排名、奧運獎牌排名、明星粉絲排名等,各大排行榜,給人的既是動力,也是壓力。

而講到排序,就會有各種排序算法和相關實現,本文不講任何排序算法,而只專注于講使用。通過實例給大家展示,我們可以了解怎樣使用既有的工具進行排序。Linux之父說:

Talk is cheap. show me the code!

本文JDK版本為Java 8,但并不代表所介紹到的所有方法只能在JDK1.8上跑,部分方法在之前的版本就已經給出。

如下本次整理的圖,記住圖中的方法,就能輕松應對大多數場景,趕緊收藏起來吧,哈哈


Java排序

兩個接口

Comparable

先上代碼:

package java.lang;

public interface Comparable<T> {  
    public int compareTo(T o);
}

可以看出這個接口只有一個方法,這個方法只有一個參數,實現了這個接口的類就可以和同類進行比較了。這個方法所實現的,就是比較法則,也是說,它表示如何對兩個對象進行比較。
它返回的是一個整數int:

  • 返回正數,代表當前對象大于被比較的對象;
  • 返回0,代表當前對象等于于被比較的對象;
  • 返回負數,代表當前對象小于被比較的對象。

實現了該接口后,我們就可以使用Arrays.sort()和Collections.sort()來進行排序了。
不然對象沒有比較法則,程序肯定是不知道如何進行比較排序的。
像我們常用的類String、Integer、Double、Date等,JDK都幫我們實現了Comparable接口,我們可以直接對這類對象進行比較排序。
舉個例子,Date Comparable的實現:

public int compareTo(Date anotherDate) {
    long thisTime = getMillisOf(this);
    long anotherTime = getMillisOf(anotherDate);
    return (thisTime<anotherTime ? -1 : (thisTime==anotherTime ? 0 : 1));
}

需要注意的是,當我們自己去實現Comparable接口時,一定要注意與equals()方法保持一致。當兩個對象是equals的,compare的結果應該是相等的。

Comparator

還是先看代碼,看看接口定義吧:

package java.util;

@FunctionalInterface
public interface Comparator<T> {   
    int compare(T o1, T o2);    
}

它是一個函數式接口,它的compare方法有兩個參數,代表進行比較的兩個對象。這個接口代表了可以作為某種對象比較的一種法則,或叫一種策略。
它的返回值正負代表意義與Comparable接口的方法一樣。
它的使用通常會有三種方式:

  1. 實現類
  2. 匿名類
  3. Lambda表達式

在Java8之后,我們一般用Lambda比較多,也比較靈活優雅。

兩個接口的比較

兩個接口功能都是用于比較排序,但其實有很大的區別。

  1. 兩者方法參數不同,Comparable只有一個參數,表示被比較的對象,因為它的方法是位于需要比較的類里的,所以只要一個參數就可以了;而Comparator的比較方法則有兩個參數,分別表示比較對象和被比較對象。
  2. Comparable與對象綁定,位于對象內,我們可以稱之為內比較器;而Comparator是獨立于需要比較的類的,我們可以稱為外比較器。
  3. 當類實現了Comparable方法后,比較法則就確定了,我們稱之為自然比較方法,我們無法給它實現多種比較方法;而因為Comparator獨立于外,我們可以為同一個類提供多種Comparator的實現,這樣來提供多種比較方法/策略,如升序倒序,因此我們也可以將Comparator看成是一種策略模式。

相對于Comparable,Comparator有一定的靈活性,假如一個類并沒有實現Comparable接口,并且這個類是無法修改的,我們就要通過提供Comparator來進行比較排序。
Comparator這種模式實現了數據與算法的解耦合,對于維護也是很方便的。


工具類

十分友好的是,JDK為我們提供了工具類,它們的靜態方法可以幫助我們直接對數組和List進行排序。

數組排序Arrays

Arrays的sort方法可以對已經實現了Comparable接口的進行排序,同時還可指定排序的范圍。

//Arrays.sort對String進行排序
String[] strings = {"de", "dc", "aA", "As", "k", "b"};
Arrays.sort(strings);
assertTrue(Arrays.equals(strings,
        new String[]{"As", "aA", "b", "dc", "de", "k"}));

指定范圍排序,需要注意的是,index是從0開始算的,包含fromIndex,不包含toIndex:

//Arrays.sort指定范圍排序
strings = new String[]{"z", "a", "d", "b"};
Arrays.sort(strings, 0, 3);
assertTrue(Arrays.equals(strings,
        new String[]{"a", "d", "z", "b"}));

對于基本類型,一樣可以進行排序,并不需要使用封裝類:

//Arrays.sort對基本類型排序
int[] nums = {3, 1, 20, 2, 38, 2, 94};
Arrays.sort(nums);
assertTrue(Arrays.equals(nums,
        new int[]{1, 2, 2, 3, 20, 38, 94}));

還能多線程進行排序,其實是拆分成多個子數組再進行排序,最終再合并結果。

//Arrays.parallelSort多線程排序
nums = new int[]{3, 1, 20, 2, 38, 2, 94};
Arrays.parallelSort(nums);
assertTrue(Arrays.equals(nums,
        new int[]{1, 2, 2, 3, 20, 38, 94}));

對于沒有實現Comparable的類也可以使用,但需要提供Comparator來指定比較策略。
本文的沒有實現Comparable接口的類Person如下:

import lombok.AllArgsConstructor;
import lombok.Data;

@Data
@AllArgsConstructor
public class Person {
    private String name;
    private int age;
}

排序:

//Arrays.sort提供Comparator進行排序
Person[] persons = new Person[]{
        new Person("Larry", 18),
        new Person("David", 30),
        new Person("James", 20),
        new Person("Harry", 18)};
Arrays.sort(persons, Comparator.comparingInt(Person::getAge));
assertTrue(Arrays.equals(persons, new Person[]{
        new Person("Larry", 18),
        new Person("Harry", 18),
        new Person("James", 20),
        new Person("David", 30)}));

List排序Collections

JDK的Collections工具類提供了排序方法,可以方便使用。
對于實現Comparable的類進行排序:

//Collections.sort對于實現Comparable的類進行排序
List<String> names = asList("Larry", "Harry", "James", "David");
Collections.sort(names);
assertEquals(names, asList("David", "Harry", "James", "Larry"));

提供Comparator進行排序:

//Collections.sort提供Comparator進行排序
List<Person> persons2 = asList(
        new Person("Larry", 18),
        new Person("David", 30),
        new Person("James", 20),
        new Person("Harry", 18));
Collections.sort(persons2, Comparator.comparingInt(Person::getAge));
assertEquals(persons2, asList(
        new Person("Larry", 18),
        new Person("Harry", 18),
        new Person("James", 20),
        new Person("David", 30)));

反序:只是把List反過來,并不代表一定是按照大小順序的:

//Collections.reverse反序
names = asList("Larry", "Harry", "James", "David");
Collections.reverse(names);
assertEquals(names, asList("David", "James", "Harry", "Larry"));

成員方法

List排序

List接口有sort(Comparator<? super E> c)方法,可以實現對自身的排序,會影響自身的順序。

//List.sort排序
names = asList("Larry", "Harry", "James", "David");
names.sort(Comparator.naturalOrder());
assertEquals(names, asList("David", "Harry", "James", "Larry"));

Stream排序

Stream提供了sorted()和sorted(Comparator<? super T> comparator)進行排序,會返回一個新的Stream。

//Stream.sorted排序
names = asList("Larry", "Harry", "James", "David");
List<String> result = names.stream()
        .sorted()
        .collect(Collectors.toList());
assertEquals(result, asList("David", "Harry", "James", "Larry"));

//Stream.sorted提供Comparator排序
names = asList("Larry", "Harry", "James", "David");
result = names.stream()
        .sorted(Comparator.naturalOrder())
        .collect(Collectors.toList());
assertEquals(result, asList("David", "Harry", "James", "Larry"));

方便對象排序的Comparator

單字段排序

對類的單字段進行排序很簡單,只要提供形如:

  • Comparator.comparing(類名::屬性getter)

的Comparator就行了。如果需要倒序,就需要:

  • Comparator.comparing(類名::屬性getter).reversed()
  • Comparator.comparing(類名::屬性getter, Comparator.reverseOrder())

具體代碼使用(為了不破壞List的原有順序,我們都使用Stream來操作):

//單字段排序-升序
List<Person> personList = asList(
        new Person("Larry", 18),
        new Person("David", 30),
        new Person("David", 3),
        new Person("James", 20),
        new Person("Harry", 18));
List<Person> personResult = personList.stream()
        .sorted(Comparator.comparing(Person::getName))
        .collect(Collectors.toList());
assertEquals(personResult, asList(
        new Person("David", 30),
        new Person("David", 3),
        new Person("Harry", 18),
        new Person("James", 20),
        new Person("Larry", 18)));

//單字段排序-倒序1
personResult = personList.stream()
        .sorted(Comparator.comparing(Person::getName).reversed())
        .collect(Collectors.toList());
assertEquals(personResult, asList(
        new Person("Larry", 18),
        new Person("James", 20),
        new Person("Harry", 18),
        new Person("David", 30),
        new Person("David", 3)));
//單字段排序-倒序2
personResult = personList.stream()
        .sorted(Comparator.comparing(Person::getName, Comparator.reverseOrder()))
        .collect(Collectors.toList());
assertEquals(personResult, asList(
        new Person("Larry", 18),
        new Person("James", 20),
        new Person("Harry", 18),
        new Person("David", 30),
        new Person("David", 3)));

多字段排序

多字段其實也很方便,只需要用thenComparing進行連接就可以:
Comparator.comparing(類名::屬性一getter).thenComparing(類名::屬性二getter)
具體代碼使用例子如下:

//多字段排序-1升2升
personResult = personList.stream()
        .sorted(Comparator.comparing(Person::getName)
                .thenComparing(Person::getAge))
        .collect(Collectors.toList());
assertEquals(personResult, asList(
        new Person("David", 3),
        new Person("David", 30),
        new Person("Harry", 18),
        new Person("James", 20),
        new Person("Larry", 18)));

//多字段排序-1升2倒
personResult = personList.stream()
        .sorted(Comparator.comparing(Person::getName)
                .thenComparing(Person::getAge, Comparator.reverseOrder()))
        .collect(Collectors.toList());
assertEquals(personResult, asList(
        new Person("David", 30),
        new Person("David", 3),
        new Person("Harry", 18),
        new Person("James", 20),
        new Person("Larry", 18)));

//多字段排序-1倒2升
personResult = personList.stream()
        .sorted(Comparator.comparing(Person::getName, Comparator.reverseOrder())
                .thenComparing(Person::getAge))
        .collect(Collectors.toList());
assertEquals(personResult, asList(
        new Person("Larry", 18),
        new Person("James", 20),
        new Person("Harry", 18),
        new Person("David", 3),
        new Person("David", 30)));

//多字段排序-1倒2倒
personResult = personList.stream()
        .sorted(Comparator.comparing(Person::getName, Comparator.reverseOrder())
                .thenComparing(Person::getAge, Comparator.reverseOrder()))
        .collect(Collectors.toList());
assertEquals(personResult, asList(
        new Person("Larry", 18),
        new Person("James", 20),
        new Person("Harry", 18),
        new Person("David", 30),
        new Person("David", 3)));

總結

本文從比較排序相關的兩個接口(Comparable和Comparator)講起,并以代碼實例的形式,講解了Array、List、Stream排序的方法,這應該可以覆蓋大部分Java排序的使用場景。
對于其它集合類如Set和Map,一樣可以進行排序處理,可以將它們轉化為Stream然后再進行排序。

?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,488評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,034評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,327評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,554評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,337評論 6 404
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,883評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,975評論 3 439
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,114評論 0 286
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,625評論 1 332
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,555評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,737評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,244評論 5 355
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 43,973評論 3 345
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,362評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,615評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,343評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,699評論 2 370

推薦閱讀更多精彩內容

  • 今日任務1、TreeSet介紹(掌握TreeSet集合的應用)2、Comparable 接口介紹(掌握)3、Com...
    Villain丶Cc閱讀 1,843評論 0 1
  • 項目中經常會遇到列表搜索查詢,大部分的查詢是可以通過sql語句來實現的,有些特殊的搜索排序sql則實現不了,例如中...
    信徒_allen閱讀 2,593評論 0 1
  • 面向對象主要針對面向過程。 面向過程的基本單元是函數。 什么是對象:EVERYTHING IS OBJECT(萬物...
    sinpi閱讀 1,070評論 0 4
  • 1.import static是Java 5增加的功能,就是將Import類中的靜態方法,可以作為本類的靜態方法來...
    XLsn0w閱讀 1,252評論 0 2
  • 一、基本數據類型 注釋 單行注釋:// 區域注釋:/* */ 文檔注釋:/** */ 數值 對于byte類型而言...
    龍貓小爺閱讀 4,278評論 0 16