Java中的Collections工具類(lèi)及Comparator和Comparable的區(qū)別

Collections類(lèi)的常見(jiàn)方法

為了處理Collection類(lèi)的實(shí)例對(duì)象巨柒,java提供了Collections工具類(lèi)來(lái)進(jìn)行操作歼指。該類(lèi)為工具類(lèi)卖丸,內(nèi)部都為static方法糠悯。來(lái)看常見(jiàn)的幾種使用:

Collections.sort()

對(duì)一個(gè)有序的List做排序帮坚。可以自定義排序規(guī)則互艾。
來(lái)看最基本的一個(gè)使用:

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

public class Main {
    public static void main(String[] args) {
        ArrayList<Integer> arr = new ArrayList<>();
        arr.add(3);
        arr.add(1);
        arr.add(4);
        arr.add(2);

        System.out.println(arr); // [3, 1, 4, 2]
        Collections.sort(arr);
        System.out.println(arr); // [1, 2, 3, 4]
    }
}

如果不指定排序規(guī)則會(huì)按照默認(rèn)排序進(jìn)行排列试和。再來(lái)看自定義規(guī)則的一個(gè)例子:

public class Main {
    public static void main(String[] args) {
        ArrayList<Integer> arr = new ArrayList<>();
        arr.add(3);
        arr.add(1);
        arr.add(4);
        arr.add(2);

        System.out.println(arr); // [3, 1, 4, 2]
        Collections.sort(arr, new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return o2 - o1;
            }
        });
        System.out.println(arr); // [4, 3, 2, 1]
    }
}

Comparator是一個(gè)接口,實(shí)現(xiàn)其內(nèi)部的compare方法即可按照給定規(guī)則排序纫普。

上述排序例子中new Comparator<Integer>() {}是Java的內(nèi)部匿名類(lèi)阅悍,本身接口是不能直接new的,這個(gè)就表示實(shí)際上現(xiàn)在new的是這個(gè)接口的實(shí)現(xiàn)類(lèi)

Comparator和Comparable的區(qū)別

上述例子中使用了Comparator节视,和前者一樣拳锚,Comparable也是一個(gè)接口,二者表示的不同的地方在于:

  • Comparable是排序接口寻行。若一個(gè)類(lèi)實(shí)現(xiàn)了Comparable接口霍掺,就意味著該類(lèi)支持排序
  • Comparator是比較接口,我們?nèi)绻枰刂颇硞€(gè)類(lèi)的次序寡痰,而該類(lèi)本身不支持排序(即沒(méi)有實(shí)現(xiàn)Comparable接口)抗楔,那么我們就可以建立一個(gè)“該類(lèi)的比較器”來(lái)進(jìn)行排序,這個(gè)“比較器”只需要實(shí)現(xiàn)Comparator接口即可拦坠。

舉個(gè)例子來(lái)說(shuō)连躏,我們這次創(chuàng)建一個(gè)自定義的可以排序的類(lèi)。

package com.DeeJay;

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

public class Main {
    public static void main(String[] args) {
        ArrayList<SortDemo> arr = new ArrayList<>();
        arr.add(new SortDemo(3));
        arr.add(new SortDemo(1));
        arr.add(new SortDemo(2));
        arr.add(new SortDemo(4));

        System.out.println(arr); // [sortDemo_3, sortDemo_1, sortDemo_2, sortDemo_4]

        Collections.sort(arr);

        System.out.println(arr); // [sortDemo_1, sortDemo_2, sortDemo_3, sortDemo_4]
    }
}
class SortDemo implements Comparable {
    int num;

    public SortDemo(int num) {
        this.num = num;
    }

    @Override
    public int compareTo(Object o) {
        SortDemo s = (SortDemo) o;
        if(this.num < s.num) {
            return -1;
        }else if(this.num > s.num) {
            return 1;
        }
        return 0;
          // 這塊不能直接寫(xiě) return this.num - s.num; 因?yàn)闀?huì)存在溢出的情況
        // return this.num - s.num;
    }

    @Override
    public String toString() {
        return "sortDemo_" + this.num;
    }
}

上述例子中贞滨,SortDemo類(lèi)實(shí)現(xiàn)了Comparable接口1入热,重寫(xiě)了compareTo方法,從而使得SortDemo類(lèi)的實(shí)例對(duì)象可以相互比較排序晓铆,直接調(diào)用Collections.sort()即可按照實(shí)現(xiàn)的compareTo方法進(jìn)行排序勺良,不需要傳入額外的比較邏輯。

對(duì)于上述情況骄噪,不能直接寫(xiě) return this.num - s.num;,舉個(gè)例子來(lái)講:byte a = -128; byte b = 1; System.out.println((byte) a - b); // 127 發(fā)生了溢出

再來(lái)看使用Comparator的例子尚困,這次我們的SortDemo類(lèi)并不支持排序,調(diào)用Collections.sort()傳入一個(gè)實(shí)現(xiàn)Comparator的內(nèi)部匿名類(lèi)來(lái)指定排序規(guī)則链蕊。

package com.DeeJay;

import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;

public class Main {
    public static void main(String[] args) {
        ArrayList<SortDemo> arr = new ArrayList<>();
        arr.add(new SortDemo(3));
        arr.add(new SortDemo(1));
        arr.add(new SortDemo(2));
        arr.add(new SortDemo(4));

        System.out.println(arr); // [sortDemo_3, sortDemo_1, sortDemo_2, sortDemo_4]

        Collections.sort(arr, new Comparator<SortDemo>() {  // 此處為內(nèi)部匿名類(lèi)
            @Override
            public int compare(SortDemo o1, SortDemo o2) {
                return o1.num - o2.num;
            }
        });

        System.out.println(arr); // [sortDemo_1, sortDemo_2, sortDemo_3, sortDemo_4]
    }
}
class SortDemo{
    int num;

    public SortDemo(int num) {
        this.num = num;
    }

    @Override
    public String toString() {
        return "sortDemo_" + this.num;
    }
}

Collections.binarySearch

返回指定List和key的index事甜。關(guān)于這個(gè)方法有地方需要注意:

要進(jìn)行查找,要先調(diào)用Collections.sort()進(jìn)行自然順序排序滔韵,不然返回的結(jié)果會(huì)有問(wèn)題
要調(diào)用Collections.sort()逻谦,那要保證List內(nèi)部的類(lèi)型是實(shí)現(xiàn)了Comparable接口的

對(duì)于Integer這種Java已經(jīng)實(shí)現(xiàn)了Comparable的類(lèi)是可以不做處理的陪蜻,但是我們自定義的類(lèi)要進(jìn)行實(shí)現(xiàn)Comparable邦马。

package com.DeeJay;

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

public class Main {
    public static void main(String[] args) {
        ArrayList<Integer> arr = new ArrayList<>();
        arr.add(3);
        arr.add(1);
        arr.add(2);
        arr.add(4);

        Collections.sort(arr); // 調(diào)用binarySearch之前要先進(jìn)行sort
        System.out.println(Collections.binarySearch(arr, 3));
    }
}

Collections.copy(destList, targetList)

這個(gè)方法需要注意的地方是,目的List的長(zhǎng)度要保證不短于要被克隆的targetList

package com.DeeJay;

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

public class Main {
    public static void main(String[] args) {
        ArrayList<Integer> arr = new ArrayList<>();
        arr.add(3);
        arr.add(1);
        arr.add(2);
        arr.add(4);
        ArrayList<Integer> arr2 = new ArrayList<>();
        arr2.add(null);
        arr2.add(null);
        arr2.add(null);
        arr2.add(null);
        Collections.copy(arr2, arr);
        System.out.println(arr2); // [3, 1, 2, 4]
    }
}

Collections.fill() Collections.reverse() Collections.shuffle()

package com.DeeJay;

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

public class Main {
    public static void main(String[] args) {
        ArrayList<Integer> arr = new ArrayList<>();
        arr.add(3);
        arr.add(1);
        arr.add(2);
        arr.add(4);
        Collections.sort(arr);
        System.out.println(arr);// [1, 2, 3, 4]
        Collections.reverse(arr);
        System.out.println(arr); // [4, 3, 2, 1]
        Collections.shuffle(arr);
        System.out.println(arr); // [1, 3, 2, 4] 隨機(jī)打亂次序
        Collections.fill(arr, null);
        System.out.println(arr); // [null, null, null, null]
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末宴卖,一起剝皮案震驚了整個(gè)濱河市滋将,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌症昏,老刑警劉巖随闽,帶你破解...
    沈念sama閱讀 212,454評(píng)論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異齿兔,居然都是意外死亡橱脸,警方通過(guò)查閱死者的電腦和手機(jī)础米,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,553評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)添诉,“玉大人屁桑,你說(shuō)我怎么就攤上這事±父埃” “怎么了蘑斧?”我有些...
    開(kāi)封第一講書(shū)人閱讀 157,921評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)须眷。 經(jīng)常有香客問(wèn)我竖瘾,道長(zhǎng),這世上最難降的妖魔是什么花颗? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,648評(píng)論 1 284
  • 正文 為了忘掉前任捕传,我火速辦了婚禮,結(jié)果婚禮上扩劝,老公的妹妹穿的比我還像新娘庸论。我一直安慰自己,他們只是感情好棒呛,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,770評(píng)論 6 386
  • 文/花漫 我一把揭開(kāi)白布聂示。 她就那樣靜靜地躺著,像睡著了一般簇秒。 火紅的嫁衣襯著肌膚如雪鱼喉。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,950評(píng)論 1 291
  • 那天趋观,我揣著相機(jī)與錄音扛禽,去河邊找鬼。 笑死拆内,一個(gè)胖子當(dāng)著我的面吹牛旋圆,可吹牛的內(nèi)容都是我干的宠默。 我是一名探鬼主播麸恍,決...
    沈念sama閱讀 39,090評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼搀矫!你這毒婦竟也來(lái)了抹沪?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,817評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤瓤球,失蹤者是張志新(化名)和其女友劉穎融欧,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體卦羡,經(jīng)...
    沈念sama閱讀 44,275評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡噪馏,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,592評(píng)論 2 327
  • 正文 我和宋清朗相戀三年麦到,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片欠肾。...
    茶點(diǎn)故事閱讀 38,724評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡瓶颠,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出刺桃,到底是詐尸還是另有隱情粹淋,我是刑警寧澤,帶...
    沈念sama閱讀 34,409評(píng)論 4 333
  • 正文 年R本政府宣布瑟慈,位于F島的核電站桃移,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏葛碧。R本人自食惡果不足惜借杰,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,052評(píng)論 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望进泼。 院中可真熱鬧第步,春花似錦、人聲如沸缘琅。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,815評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)刷袍。三九已至翩隧,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間呻纹,已是汗流浹背堆生。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,043評(píng)論 1 266
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留雷酪,地道東北人淑仆。 一個(gè)月前我還...
    沈念sama閱讀 46,503評(píng)論 2 361
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像哥力,于是被迫代替她去往敵國(guó)和親蔗怠。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,627評(píng)論 2 350

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

  • 項(xiàng)目中經(jīng)常會(huì)遇到列表搜索查詢(xún)吩跋,大部分的查詢(xún)是可以通過(guò)sql語(yǔ)句來(lái)實(shí)現(xiàn)的寞射,有些特殊的搜索排序sql則實(shí)現(xiàn)不了,例如中...
    信徒_allen閱讀 2,580評(píng)論 0 1
  • 一、基本數(shù)據(jù)類(lèi)型 注釋 單行注釋?zhuān)?/ 區(qū)域注釋?zhuān)?* */ 文檔注釋?zhuān)?** */ 數(shù)值 對(duì)于byte類(lèi)型而言...
    龍貓小爺閱讀 4,257評(píng)論 0 16
  • java筆記第一天 == 和 equals ==比較的比較的是兩個(gè)變量的值是否相等梁丘,對(duì)于引用型變量表示的是兩個(gè)變量...
    jmychou閱讀 1,490評(píng)論 0 3
  • Java 語(yǔ)言支持的類(lèi)型分為兩類(lèi):基本類(lèi)型和引用類(lèi)型侵浸。整型(byte 1, short 2, int 4, lon...
    xiaogmail閱讀 1,346評(píng)論 0 10
  • 我們可以回顧一下在《紅樓夢(mèng)》第七十七回“俏丫鬟抱屈夭風(fēng)流”中的一些非常關(guān)鍵的文字:寶玉哭道:“我究竟不知晴雯犯了何...
    半夏畏姜閱讀 232評(píng)論 0 3