數(shù)據(jù)結(jié)構(gòu)-優(yōu)先隊(duì)列(priority-queue)

在刷leetcode算法題的時(shí)候偶然間發(fā)現(xiàn)了一種非常好用的數(shù)據(jù)結(jié)構(gòu)——優(yōu)先隊(duì)列,與普通隊(duì)列不同的是它會(huì)在你插入時(shí)就幫你根據(jù)優(yōu)先級對數(shù)據(jù)進(jìn)行排序,底層實(shí)現(xiàn)是用了堆排序奕纫。
在很多語言中其實(shí)都有內(nèi)置(或者在常用庫中有)這種數(shù)據(jù)結(jié)構(gòu)署惯,如果是使用js的話是需要先安裝對應(yīng)的依賴

npm install --save @datastructures-js/priority-queue

引入方式

import {
  PriorityQueue, // 優(yōu)先隊(duì)列又官,可自定義排序方式延刘,較為靈活
  MinPriorityQueue, // 最大優(yōu)先隊(duì)列
  MaxPriorityQueue, // 最小優(yōu)先隊(duì)列
  ICompare, // 比較的方法的類型
  IGetCompareValue, // 比較的值的類型
} from '@datastructures-js/priority-queue';

PriorityQueue

首先要介紹的就是優(yōu)先隊(duì)列這個(gè)類,創(chuàng)建它的實(shí)例時(shí)需要傳入一個(gè)函數(shù)六敬,類似于sort的回調(diào)函數(shù)碘赖,返回大于0的數(shù)表示兩項(xiàng)需要互換位置。

interface ICar {
  year: number;
  price: number;
}

// 這個(gè)比較函數(shù)表示年份最大(最新)優(yōu)先外构,若相等則價(jià)格最小優(yōu)先
const compareCars: ICompare<ICar> = (a: ICar, b: ICar) => {
  if (a.year > b.year) {
    return -1;
  }
  if (a.year < b.year) {
    return 1;
  }
  return a.price < b.price ? -1 : 1;
};

const carsQueue = new PriorityQueue<ICar>(compareCars);

MinPriorityQueue, MaxPriorityQueue

最大普泡、最小優(yōu)先隊(duì)列則不需要傳入一個(gè)比較函數(shù),只需要指定進(jìn)行比較的對象即可审编,若元素為數(shù)字劫哼、字符串等原始變量,創(chuàng)建時(shí)不需要傳入比較函數(shù)

const numbersQueue = new MinPriorityQueue<number>();

interface IBid {
  id: number;
  value: number;
}
const getBidValue: IGetCompareValue<IBid> = (bid) => bid.value;
const bidsQueue = new MaxPriorityQueue<IBid>(getBidValue);

const numbersQueue = new MinPriorityQueue(); // 不傳則直接比較元素割笙,最小優(yōu)先

fromArray

fromArray是這三個(gè)類上的一個(gè)方法,可以在O(n)的時(shí)間復(fù)雜度上將一個(gè)數(shù)組轉(zhuǎn)化為優(yōu)先隊(duì)列這種數(shù)據(jù)結(jié)構(gòu):

PriorityQueue
const numbers = [3, -2, 5, 0, -1, -5, 4];

const pq = PriorityQueue.fromArray<number>(numbers, (a, b) => a - b);

console.log(numbers); // [-5, -1, -2, 3, 0, 5, 4]
pq.dequeue(); // -5
pq.dequeue(); // -2
pq.dequeue(); // -1
console.log(numbers); // [ 0, 3, 4, 5 ]
MinPriorityQueue, MaxPriorityQueue
const numbers = [3, -2, 5, 0, -1, -5, 4];

const mpq = MaxPriorityQueue.fromArray<number>(numbers);

console.log(numbers); // [-5, -1, -2, 3, 0, 5, 4]
mpq.dequeue(); // 5
mpq.dequeue(); // 4
mpq.dequeue(); // 3
console.log(numbers); // [ 0, -1, -5, -2 ]

enqueue (push)

這三個(gè)類中插入元素的方法眯亦,使用enqueue(push是別名伤溉,效果一樣),可以以O(shè)(log(n))的復(fù)雜度插入數(shù)據(jù):

const cars = [
  { year: 2013, price: 35000 },
  { year: 2010, price: 2000 },
  { year: 2013, price: 30000 },
  { year: 2017, price: 50000 },
  { year: 2013, price: 25000 },
  { year: 2015, price: 40000 },
  { year: 2022, price: 70000 }
];
cars.forEach((car) => carsQueue.enqueue(car));

const numbers = [3, -2, 5, 0, -1, -5, 4];
numbers.forEach((num) => numbersQueue.push(num)); // push is an alias for enqueue

const bids = [
  { id: 1, value: 1000 },
  { id: 2, value: 20000 },
  { id: 3, value: 1000 },
  { id: 4, value: 1500 },
  { id: 5, value: 12000 },
  { id: 6, value: 4000 },
  { id: 7, value: 8000 }
];
bids.forEach((bid) => bidsQueue.enqueue(bid));

front

front方法可以獲取當(dāng)前隊(duì)列中優(yōu)先級最高的那一項(xiàng):

console.log(carsQueue.front()); // { year: 2022, price: 70000 }
console.log(numbersQueue.front()); // -5
console.log(bidsQueue.front()); // { id: 2, value: 20000 }

back

back方法可以獲取當(dāng)前隊(duì)列中優(yōu)先級最低的那一項(xiàng):

console.log(carsQueue.back()); // { year: 2010, price: 2000 }
console.log(numbersQueue.back()); // 5
console.log(bidsQueue.back()); // { id: 1, value: 1000 }

dequeue (pop)

dequeue(別名pop)方法的效果是返回并移除隊(duì)列中優(yōu)先級最高的一項(xiàng)(即出列)妻率,時(shí)間復(fù)雜度同樣是O(log(n)):

console.log(carsQueue.dequeue()); // { year: 2022, price: 70000 }
console.log(carsQueue.dequeue()); // { year: 2017, price: 50000 }
console.log(carsQueue.dequeue()); // { year: 2015, price: 40000 }

console.log(numbersQueue.dequeue()); // -5
console.log(numbersQueue.dequeue()); // -2
console.log(numbersQueue.dequeue()); // -1

console.log(bidsQueue.pop()); // { id: 2, value: 20000 }
console.log(bidsQueue.pop()); // { id: 5, value: 12000 }
console.log(bidsQueue.pop()); // { id: 7, value: 8000 }

remove

remove方法需要傳入一個(gè)比較函數(shù)乱顾,效果是返回并移除符合條件的項(xiàng),時(shí)間復(fù)雜度是O(n*log(n)):

carsQueue.remove((car) => car.price === 35000); // [{ year: 2013, price: 35000 }]

numbersQueue.remove((n) => n === 4); // [4]

bidsQueue.remove((bid) => bid.id === 3); // [{ id: 3, value: 1000 }]

isEmpty

isEmpty方法可以用來判斷隊(duì)列是否為空:

console.log(carsQueue.isEmpty()); // false
console.log(numbersQueue.isEmpty()); // false
console.log(bidsQueue.isEmpty()); // false

size

size方法可以用來返回隊(duì)列中項(xiàng)的個(gè)數(shù):

console.log(carsQueue.size()); // 3
console.log(numbersQueue.size()); // 3
console.log(bidsQueue.size()); // 3

toArray

toArray方法可以在O(n*log(n))的時(shí)間復(fù)雜度下將優(yōu)先隊(duì)列數(shù)據(jù)結(jié)構(gòu)轉(zhuǎn)為普通的數(shù)組:

console.log(carsQueue.toArray());
/*
[
  { year: 2013, price: 25000 },
  { year: 2013, price: 30000 },
  { year: 2010, price: 2000 }
]
*/

console.log(numbersQueue.toArray()); // [ 0, 3, 5 ]

console.log(bidsQueue.toArray());
/*
[
  { id: 6, value: 4000 },
  { id: 4, value: 1500 },
  { id: 1, value: 1000 }
]
*/

clear

clear方法可以清空整個(gè)隊(duì)列:

carsQueue.clear();
console.log(carsQueue.size()); // 0
console.log(carsQueue.front()); // null
console.log(carsQueue.dequeue()); // null
console.log(carsQueue.isEmpty()); // true

numbersQueue.clear();
console.log(numbersQueue.size()); // 0
console.log(numbersQueue.front()); // null
console.log(numbersQueue.dequeue()); // null
console.log(numbersQueue.isEmpty()); // true

bidsQueue.clear();
console.log(bidsQueue.size()); // 0
console.log(bidsQueue.front()); // null
console.log(bidsQueue.dequeue()); // null
console.log(bidsQueue.isEmpty()); // true

Symbol.iterator

優(yōu)先隊(duì)列這一數(shù)據(jù)結(jié)構(gòu)也實(shí)現(xiàn)Symbol.iterator這一遍歷器接口宫静,可以進(jìn)行相關(guān)展開走净、循環(huán)操作:

console.log([...carsQueue]);
/*
[
  { year: 2013, price: 25000 },
  { year: 2013, price: 30000 },
  { year: 2010, price: 2000 }
]
*/
console.log(carsQueue.size()); // 0

console.log([...numbersQueue]); // [ 0, 3, 5 ]
console.log(numbersQueue.size()); // 0

for (const bid of bidsQueue) {
  console.log(bid);
}
/*
{ id: 6, value: 4000 },
{ id: 4, value: 1500 },
{ id: 1, value: 1000 }
*/
console.log(bidsHeap.size()); // 0

使用場景

奉上一個(gè)LeetCode題目(2208. 將數(shù)組和減半的最少操作次數(shù)):

var halveArray = function(nums) {
    const pq = new MaxPriorityQueue();
    for (const num of nums) {
        pq.enqueue(num);
    }
    let res = 0;
    let sum1 = nums.reduce((acc, curr) => acc + curr, 0);
    let sum2 = 0;
    while (sum2 < sum1 / 2) {
        const x = pq.dequeue().element;
        sum2 += x / 2;
        pq.enqueue(x / 2);
        res++;
    }
    return res;
};
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市孤里,隨后出現(xiàn)的幾起案子伏伯,更是在濱河造成了極大的恐慌,老刑警劉巖捌袜,帶你破解...
    沈念sama閱讀 216,591評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件说搅,死亡現(xiàn)場離奇詭異,居然都是意外死亡虏等,警方通過查閱死者的電腦和手機(jī)弄唧,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,448評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來霍衫,“玉大人候引,你說我怎么就攤上這事《氐” “怎么了澄干?”我有些...
    開封第一講書人閱讀 162,823評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我傻寂,道長息尺,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,204評論 1 292
  • 正文 為了忘掉前任疾掰,我火速辦了婚禮搂誉,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘静檬。我一直安慰自己炭懊,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,228評論 6 388
  • 文/花漫 我一把揭開白布拂檩。 她就那樣靜靜地躺著侮腹,像睡著了一般。 火紅的嫁衣襯著肌膚如雪稻励。 梳的紋絲不亂的頭發(fā)上父阻,一...
    開封第一講書人閱讀 51,190評論 1 299
  • 那天,我揣著相機(jī)與錄音望抽,去河邊找鬼加矛。 笑死,一個(gè)胖子當(dāng)著我的面吹牛煤篙,可吹牛的內(nèi)容都是我干的斟览。 我是一名探鬼主播,決...
    沈念sama閱讀 40,078評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼辑奈,長吁一口氣:“原來是場噩夢啊……” “哼苛茂!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起鸠窗,我...
    開封第一講書人閱讀 38,923評論 0 274
  • 序言:老撾萬榮一對情侶失蹤妓羊,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后稍计,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體侍瑟,經(jīng)...
    沈念sama閱讀 45,334評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,550評論 2 333
  • 正文 我和宋清朗相戀三年丙猬,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了涨颜。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,727評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡茧球,死狀恐怖庭瑰,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情抢埋,我是刑警寧澤弹灭,帶...
    沈念sama閱讀 35,428評論 5 343
  • 正文 年R本政府宣布督暂,位于F島的核電站,受9級特大地震影響穷吮,放射性物質(zhì)發(fā)生泄漏逻翁。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,022評論 3 326
  • 文/蒙蒙 一捡鱼、第九天 我趴在偏房一處隱蔽的房頂上張望八回。 院中可真熱鬧,春花似錦驾诈、人聲如沸缠诅。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,672評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽管引。三九已至,卻和暖如春闯两,著一層夾襖步出監(jiān)牢的瞬間褥伴,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,826評論 1 269
  • 我被黑心中介騙來泰國打工漾狼, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留噩翠,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,734評論 2 368
  • 正文 我出身青樓邦投,卻偏偏與公主長得像,于是被迫代替她去往敵國和親擅笔。 傳聞我的和親對象是個(gè)殘疾皇子志衣,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,619評論 2 354

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