JDK8?中Arrays.sort()?排序方法詳解
一、引言
在刷算法的時(shí)候經(jīng)常需要對(duì)數(shù)組
進(jìn)行排序,第一反應(yīng)就是直接使用java.util包下的Arrays.sort()方法直接排序。但在刷算法時(shí)會(huì)通過(guò)時(shí)間復(fù)雜度
和空間復(fù)雜度
對(duì)實(shí)現(xiàn)的算法進(jìn)行評(píng)價(jià),因此我們需對(duì)Arrays.sort()方法有所了解。
本文先行介紹Arrays.sort()中影響排序方式的幾個(gè)因素。影響因素主要為數(shù)組類(lèi)型
、數(shù)組大小
,結(jié)合閾值
對(duì)排序方式進(jìn)行選擇。
二、Arrays.sort()支持類(lèi)型
Arrays.sort()重載了很多方法,支持多種數(shù)據(jù)類(lèi)型的排序。
三、核心方法DualPivotQuicksort.sort()
進(jìn)入Arrays.sort()方法的源碼,發(fā)現(xiàn)內(nèi)部主要通過(guò)DualPivotQuicksort.sort()方法實(shí)現(xiàn)排序。該方法通過(guò)數(shù)組大小、類(lèi)型結(jié)合幾個(gè)閾值來(lái)決定使用哪種排序方式。
public static void sort(int[] a) { DualPivotQuicksort.sort(a, 0, a.length - 1, null, 0, 0); }
DualPivotQuicksort類(lèi)中的幾個(gè)常量都是比較關(guān)鍵的閾值,決定了該數(shù)組的排序使用哪種方法排序。
/** * 長(zhǎng)度小于286的數(shù)組,優(yōu)先使用快排而不是歸并 */ private static final int QUICKSORT_THRESHOLD = 286; /** * 長(zhǎng)度小于47的數(shù)組,優(yōu)先使用插入而不是快排 */ private static final int INSERTION_SORT_THRESHOLD = 47; /** * 如果是byte數(shù)組,長(zhǎng)度大于29,計(jì)數(shù)排序優(yōu)先于插入排序 */ private static final int COUNTING_SORT_THRESHOLD_FOR_BYTE = 29; /** * 如果是char數(shù)組,長(zhǎng)度大于3200,計(jì)數(shù)排序優(yōu)先于快排 */ private static final int COUNTING_SORT_THRESHOLD_FOR_SHORT_OR_CHAR = 3200;
1、一般情況的排序方法選擇
簡(jiǎn)單來(lái)說(shuō),會(huì)先計(jì)算需要排序的數(shù)組長(zhǎng)度為n,再根據(jù)n的大小及數(shù)組元素類(lèi)型來(lái)決定使用什么排序。
根據(jù)前兩個(gè)閾值QUICKSORT_THRESHOLD(286)和INSERTION_SORT_THRESHOLD(47),我們可以看到大多數(shù)情況下,排序方法的使用規(guī)則是這樣的,我們規(guī)定需要排序的數(shù)組長(zhǎng)度為n。
- n < 47,使用插入排序
- 47 <= n < 286,使用快速排序
- n >= 286,使用歸并排序
2、byte、char類(lèi)型的排序
但是,當(dāng)數(shù)組類(lèi)型為byte或者char時(shí),會(huì)使用到其他兩個(gè)閾值
數(shù)組類(lèi)型為byte
時(shí),查看源碼,當(dāng)數(shù)組長(zhǎng)度n(right - left) > 29 (COUNTING_SORT_THRESHOLD_FOR_BYTE),使用計(jì)數(shù)排序
,反之,在小數(shù)組的情況下使用插入排序
static void sort(byte[] a, int left, int right) { // Use counting sort on large arrays if (right - left > COUNTING_SORT_THRESHOLD_FOR_BYTE) { int[] count = new int[NUM_BYTE_VALUES]; ... } else { // Use insertion sort on small arrays } }
數(shù)組類(lèi)型為char
時(shí),查看源碼實(shí)現(xiàn),當(dāng)數(shù)組長(zhǎng)度n(right - left) < 3200 (COUNTING_SORT_THRESHOLD_FOR_SHORT_OR_CHAR ) ,使用計(jì)數(shù)排序
,反之,使用雙軸快排
。
static void sort(char[] a, int left, int right, char[] work, int workBase, int workLen) { // Use counting sort on large arrays if (right - left > COUNTING_SORT_THRESHOLD_FOR_SHORT_OR_CHAR) { int[] count = new int[NUM_CHAR_VALUES]; ... } else { // Use Dual-Pivot Quicksort on small arrays doSort(a, left, right, work, workBase, workLen); } }
到此這篇關(guān)于JDK8 中Arrays.sort() 排序方法詳解的文章就介紹到這了,更多相關(guān)JDK8 Arrays.sort() 排序內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Springboot+WebSocket實(shí)現(xiàn)一對(duì)一聊天和公告的示例代碼
這篇文章主要介紹了Springboot+WebSocket實(shí)現(xiàn)一對(duì)一聊天和公告的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04JAVA 獲取系統(tǒng)當(dāng)前時(shí)間實(shí)例代碼
這篇文章主要介紹了JAVA 獲取系統(tǒng)當(dāng)前時(shí)間實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2016-10-10Java正則表達(dá)式之split()方法實(shí)例詳解
這篇文章主要介紹了Java正則表達(dá)式之split()方法,結(jié)合實(shí)例形式較為詳細(xì)的分析了split方法的功能、使用方法及相關(guān)注意事項(xiàng),需要的朋友可以參考下2017-03-03Java實(shí)現(xiàn)將Word轉(zhuǎn)換成Html的示例代碼
在業(yè)務(wù)中,常常會(huì)需要在瀏覽器中預(yù)覽Word文檔,或者需要將Word文檔轉(zhuǎn)成HTML文件保存,本文主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)Word轉(zhuǎn)換成Html的相關(guān)方法,希望對(duì)大家有所幫助2024-02-02關(guān)于SpringBoot改動(dòng)后0.03秒啟動(dòng)的問(wèn)題
這篇文章主要介紹了SpringBoot改動(dòng)后0.03秒啟動(dòng),本文結(jié)合示例代碼給大家講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-12-12Java中.divide()方法使用及注意事項(xiàng)詳解
divide方法就是bigdecimal類(lèi)中的一個(gè)除法計(jì)算方法,由于該divide方法參數(shù)類(lèi)型眾多并且不易理解容易出現(xiàn)錯(cuò)誤,這篇文章主要給大家介紹了關(guān)于Java中.divide()方法使用及注意事項(xiàng)的相關(guān)資料,需要的朋友可以參考下2024-03-03java 如何把byte轉(zhuǎn)化為KB、MB、GB的方法
這篇文章主要介紹了java 如何把byte轉(zhuǎn)化為KB、MB、GB的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-10-10