怎么在Java中實(shí)現(xiàn)堆排序算法?針對(duì)這個(gè)問(wèn)題,這篇文章詳細(xì)介紹了相對(duì)應(yīng)的分析和解答,希望可以幫助更多想解決這個(gè)問(wèn)題的小伙伴找到更簡(jiǎn)單易行的方法。
成都創(chuàng)新互聯(lián)是一家成都做網(wǎng)站、成都網(wǎng)站制作,提供網(wǎng)頁(yè)設(shè)計(jì),網(wǎng)站設(shè)計(jì),網(wǎng)站制作,建網(wǎng)站,定制開(kāi)發(fā),網(wǎng)站開(kāi)發(fā)公司,于2013年創(chuàng)立是互聯(lián)行業(yè)建設(shè)者,服務(wù)者。以提升客戶(hù)品牌價(jià)值為核心業(yè)務(wù),全程參與項(xiàng)目的網(wǎng)站策劃設(shè)計(jì)制作,前端開(kāi)發(fā),后臺(tái)程序制作以及后期項(xiàng)目運(yùn)營(yíng)并提出專(zhuān)業(yè)建議和思路。
什么是頂堆?
它是一顆完全二叉樹(shù),頂堆有大頂堆和小頂堆兩種。所謂大頂堆就是在這顆完全二叉樹(shù)中,任何一顆子樹(shù)都滿(mǎn)足:父結(jié)點(diǎn)的值 > 孩子結(jié)點(diǎn)的值;小頂堆則相反。
如圖:
什么是堆排序(Heapsort)?
利用堆積樹(shù)(堆)這種數(shù)據(jù)結(jié)構(gòu)所設(shè)計(jì)的一種排序算法,它是選擇排序的一種??梢岳脭?shù)組的特點(diǎn)快速定位指定索引的元素。
現(xiàn)在給我們一個(gè)無(wú)序數(shù)組,我們將其從小到大排序,使用堆排序的實(shí)現(xiàn)步驟和思想如下:
1.讓這個(gè)數(shù)組變成一個(gè)大根堆
2.將最后一個(gè)位置和堆頂位置作交換
3.將堆的大小進(jìn)行 -1 操作
4.將當(dāng)前的堆變成一個(gè)大頂堆
5.回到2
看起來(lái)似乎很抽象,那我們現(xiàn)在用一個(gè)例子進(jìn)行講解:假設(shè)一個(gè)整型數(shù)組arr = {2, 1, 3, 6, 0, 5}
1.將一個(gè)無(wú)序數(shù)組變成一個(gè)大頂堆存儲(chǔ)
如果使用完全二叉樹(shù)進(jìn)行存儲(chǔ)數(shù)組,任意下標(biāo)為index的結(jié)點(diǎn)的父結(jié)點(diǎn)的下標(biāo)應(yīng)該是(index-1)/2,其孩子結(jié)點(diǎn)的下標(biāo)應(yīng)分別為:(index * 2 + 1) 和 (index * 2 + 2) 。我們使用該結(jié)論創(chuàng)建一個(gè)大頂堆:
首先我們假設(shè)0位置上的數(shù)已經(jīng)排好,將其放在這棵二叉樹(shù)的根位置,創(chuàng)建一個(gè)int類(lèi)型變量 i 記錄當(dāng)前指向的數(shù)組的下標(biāo),初始化值為1。
設(shè)置一個(gè)index初始化值 = i,將index和(index-1)/2位置的數(shù)進(jìn)行比較,也就是和它的父結(jié)點(diǎn)進(jìn)行比較,如果比父結(jié)點(diǎn)小就不變,并進(jìn)行 i++,index = i;如果比父結(jié)點(diǎn)大就和父結(jié)點(diǎn)交換并且給index賦值為(index-1)/2,即指向原來(lái)位置的父結(jié)點(diǎn),再將該值與當(dāng)前結(jié)點(diǎn)的父結(jié)點(diǎn)進(jìn)行比較…直到該結(jié)點(diǎn)值是小于該結(jié)點(diǎn)父結(jié)點(diǎn)的值或到根結(jié)點(diǎn)時(shí)停止。
以arr數(shù)組進(jìn)行舉例:
0位置上的數(shù)是2,先認(rèn)為它是已經(jīng)排好的,i 和 index此時(shí)都為1,(index - 1)/2為0,所以將1和2進(jìn)行比較,1 < 2 所以1位置上的數(shù)不變,執(zhí)行 i++, index = i;
此時(shí)i 和 index值都為2,(index - 1)/2為0,所以講3 和 2進(jìn)行比較,3 > 2所以將3和2進(jìn)行交換,原數(shù)組就變?yōu)椋簕3, 1, 2, 6, 0, 5},index = (index - 1)/2 = 0,當(dāng)前結(jié)點(diǎn)是根節(jié)點(diǎn),不再進(jìn)行比較了,執(zhí)行 i++, index = i;
此時(shí)i 和 index值都為3,(index - 1)/2為 1,所以將 6 和 1 進(jìn)行比較,6 > 1所以將6和2進(jìn)行交換,原數(shù)組就變?yōu)椋簕3, 6, 2, 1, 0, 5},index = (index - 1)/2 = 1,不是根節(jié)點(diǎn),于是再將6 和 3進(jìn)行比較,6 > 3,所以再交換6 和 3,原數(shù)組變?yōu)椋簕6, 3, 2, 1, 0, 5},index = (index - 1)/2 = 0,當(dāng)前結(jié)點(diǎn)是根節(jié)點(diǎn),不再進(jìn)行比較了,執(zhí)行 i++, index = i;
…….
以此類(lèi)推最后得到的數(shù)組:{6, 3, 5, 1, 0, 2}
最后得到的大頂堆:
代碼實(shí)現(xiàn):
// 插入數(shù)使其形成大頂堆 public static void heapInsert(int[] arr, int index) { while(arr[index] > arr[(index - 1)/2]) { swap(arr, index, (index - 1)/2); index = (index - 1)/2; } }
2.將形成的大頂堆最后一個(gè)元素和根進(jìn)行交換
在該例子中應(yīng)該是將2和6進(jìn)行交換,得到:
得到的數(shù)組:{2, 3, 5, 1, 0, 6}
那我們?yōu)槭裁匆M(jìn)行交換呢?
這時(shí)候的數(shù)組最后一個(gè)值就是最大的了,也就是最后一個(gè)位置上的數(shù)已經(jīng)排好了,接下來(lái)我們就要將除了最后位置的結(jié)點(diǎn)之外剩下的完全二叉樹(shù)進(jìn)行排序了,即:
要進(jìn)行排序的部分:{2, 3, 5, 1, 0}
3.將除了最后一個(gè)結(jié)點(diǎn)剩下的完全二叉樹(shù)轉(zhuǎn)化成一個(gè)新的大頂堆
傳入當(dāng)前數(shù)組,并標(biāo)記當(dāng)前位置index,初始化值為0,先判斷當(dāng)前的index位置的結(jié)點(diǎn)是否是葉子結(jié)點(diǎn),如果是葉子結(jié)點(diǎn)就不需要再比較了,直接返回;如果不是葉子結(jié)點(diǎn),則將index位置的值和它孩子結(jié)點(diǎn)的值進(jìn)行比較,如果index位置上的值最大則不交換并且直接返回,否則選取最大的值與index位置上的值進(jìn)行交換;交換后index為當(dāng)前位置,再與當(dāng)前位置的孩子結(jié)點(diǎn)進(jìn)行比較。。。以此類(lèi)推直到當(dāng)前結(jié)點(diǎn)是一個(gè)葉子結(jié)點(diǎn)或不需要再交換時(shí)停止。
該例子中:index位置上的值是2,該位置的孩子結(jié)點(diǎn)分別為3 和 5,將2和5進(jìn)行交換之后,index = index * 2 + 2 = 2,此時(shí)index位置結(jié)點(diǎn)是一個(gè)葉子結(jié)點(diǎn),不再進(jìn)行交換,此時(shí)構(gòu)成了一個(gè)新的大頂堆。如圖:
得到的數(shù)組:{5, 3, 2, 1, 0, 6}
然后就回到了步驟2,直到數(shù)組中未排序的部分只有一個(gè)數(shù)時(shí)不再進(jìn)行排序。
完整代碼實(shí)現(xiàn):
/** * @author LZD 2018/03/01 */ public class HeapSort { public static void heapSort(int[] arr) { if(arr == null || arr.length < 2) return; // 構(gòu)建大頂堆 for(int i = 0;i < arr.length;i++) { heapInsert(arr, i); } int heapSize = arr.length; swap(arr, 0, --heapSize); while(heapSize > 0) { heapify(arr, 0, heapSize); swap(arr, 0, --heapSize); } } // 插入數(shù)使其形成大頂堆 public static void heapInsert(int[] arr, int index) { while(arr[index] > arr[(index - 1)/2]) { swap(arr, index, (index - 1)/2); index = (index - 1)/2; } } // 將堆化為大頂堆 public static void heapify(int[] arr, int index, int heapSize) { // 先判斷當(dāng)前的index位置的結(jié)點(diǎn)是否是葉子結(jié)點(diǎn) int left = index * 2 + 1; while(left < heapSize) { // 不是葉子結(jié)點(diǎn)則選出index位置結(jié)點(diǎn)的孩子結(jié)點(diǎn)中較大的賦給largest int largest = left+1 < heapSize && arr[left + 1] > arr[left] ? left + 1: left; largest = arr[largest] > arr[index] ? largest : index; if(largest == index) { break; } swap(arr, largest, index); index = largest; left = index * 2 + 1; } } public static void swap(int[] arr, int i, int j) { int tmp = arr[i]; arr[i] = arr[j]; arr[j] = tmp; } // for test 打印數(shù)組 public static void printArray(int[] arr) { for(int i = 0;i < arr.length;i++) { System.out.print(arr[i] + " "); } System.out.println(); } // for test public static void main(String[] args) { int[] arr = {2, 1, 3, 6, 0, 5}; heapSort(arr); printArray(arr); } }
關(guān)于怎么在Java中實(shí)現(xiàn)堆排序算法問(wèn)題的解答就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,如果你還有很多疑惑沒(méi)有解開(kāi),可以關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道了解更多相關(guān)知識(shí)。