java優(yōu)先隊列PriorityQueue中Comparator的用法詳解
在使用java的優(yōu)先隊列PriorityQueue的時候,會看到這樣的用法。
PriorityQueue<Integer> queue = new PriorityQueue<Integer>(new Comparator<Integer>(){
@Override
public int compare(Integer o1, Integer o2){
return o1.compareTo(o2);
}
});
那這樣到底構(gòu)造的是最大優(yōu)先還是最小優(yōu)先隊列呢?
看看源碼
看看offer(我也想要offer:X):
public boolean offer(E e) {
if (e == null) {
throw new NullPointerException();
} else {
++this.modCount;
int i = this.size;
if (i >= this.queue.length) {
this.grow(i + 1);
}
this.siftUp(i, e);
this.size = i + 1;
return true;
}
}
1)if和else,分別執(zhí)行對象判空和容量判斷
2)執(zhí)行siftUp(i, e),i是原有隊列長度,e是要入隊的元素。
siftUp是堆中調(diào)整元素位置的一種方法,可以看出這里的優(yōu)先隊列是使用最大/最小堆實現(xiàn)的。接著看siftUp:
private void siftUp(int k, E x) {
if (this.comparator != null) {
siftUpUsingComparator(k, x, this.queue, this.comparator);
} else {
siftUpComparable(k, x, this.queue);
}
}
看看使用了comparator的方法,k是原有隊列長度,x是入隊元素,queue是隊列,comparator是比較器:
private static <T> void siftUpUsingComparator(int k, T x, Object[] es, Comparator<? super T> cmp) {
while(true) {
if (k > 0) {
int parent = k - 1 >>> 1;
Object e = es[parent];
if (cmp.compare(x, e) < 0) {
es[k] = e;
k = parent;
continue;
}
}
es[k] = x;
return;
}
}
1)k>0,隊列長度大于0
2)parent = k - 1 >>> 1; 即(k-1)/2,表示最后一個非葉子節(jié)點的位置
3)e為父節(jié)點,x是入隊元素,x可以看做放在最后一個位置。如果compare(x, e) < 0,則執(zhí)行元素往上走的方法。
注:在siftUp中,如果是最小堆,那么應該是較小的元素往上走,如果是最大堆,則應該是較大的元素往上走。
由于源碼中新入隊元素x是在第1個參數(shù)的位置,因此最大/最小優(yōu)先隊列主要根據(jù)第1個參數(shù)的大小關(guān)系來判斷。
//對于最大堆,當x>e時,讓x上升,則 x>e時返回負數(shù),即
int compare(Integer x, Integer e){
return x > e ? -1 : 1;
}
//對于最小堆,當x<e時,讓compare(x, e) < 0,即
int compare(Integer x, Integer e){
return x < e ? -1 : 1; // return x.compareTo(e);
}
結(jié)論:
// 最小優(yōu)先隊列,直接 return o1.compareTo(o2);
PriorityQueue<Integer> queue = new PriorityQueue<Integer>(new Comparator<Integer>(){
@Override
public int compare(Integer o1, Integer o2){
return o1 < o2 ? -1 : 1;
/* e.g., return o1.compare(o2); */
}
});
// 最大優(yōu)先隊列,則反過來 return o2.compareTo(o1);
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
- 一口氣說出Java 6種延時隊列的實現(xiàn)方法(面試官也得服)
- 詳解Java阻塞隊列(BlockingQueue)的實現(xiàn)原理
- java中棧和隊列的實現(xiàn)和API的用法(詳解)
- Java 隊列實現(xiàn)原理及簡單實現(xiàn)代碼
- 解析Java中PriorityQueue優(yōu)先級隊列結(jié)構(gòu)的源碼及用法
- 剖析Java中阻塞隊列的實現(xiàn)原理及應用場景
- Java利用Redis實現(xiàn)消息隊列的示例代碼
- java實現(xiàn)消息隊列的兩種方式(小結(jié))
- 詳解Java消息隊列-Spring整合ActiveMq
- Java中和隊列相關(guān)的基本操作
相關(guān)文章
spring注解之@Valid和@Validated的區(qū)分總結(jié)
@Validated和@Valid在基本驗證功能上沒有太多區(qū)別,但在分組、注解地方、嵌套驗證等功能上有所不同,下面這篇文章主要給大家介紹了關(guān)于spring注解之@Valid和@Validated區(qū)分的相關(guān)資料,需要的朋友可以參考下2022-03-03
關(guān)于maven環(huán)境的安裝及maven集成idea環(huán)境的問題
Maven 是一個基于 Java 的工具,所以要做的第一件事情就是安裝 JDK。本文重點給大家介紹關(guān)于maven環(huán)境的安裝及和idea環(huán)境的集成問題,感興趣的朋友一起看看吧2021-09-09
Spring Cloud超詳細i講解Feign自定義配置與使用
這篇文章主要介紹了SpringCloud Feign自定義配置與使用,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-06-06
Java基于正則表達式實現(xiàn)查找匹配的文本功能【經(jīng)典實例】
這篇文章主要介紹了Java基于正則表達式實現(xiàn)查找匹配的文本功能,結(jié)合具體實例形式分析了java正則查找、字符串遍歷、group分組相關(guān)操作技巧,需要的朋友可以參考下2017-04-04
springboot如何獲取yml文件的自定義參數(shù)
這篇文章主要介紹了springboot如何獲取yml文件的自定義參數(shù),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-03-03

