詳解堆的javascript實現(xiàn)方法
堆的定義
最大(最?。┒咽且豢妹恳粋€節(jié)點的鍵值都不小于(大于)其孩子(如果存在)的鍵值的樹。大頂堆是一棵完全二叉樹,同時也是一棵最大樹。小頂堆是一棵完全完全二叉樹,同時也是一棵最小樹。
另外,記住這兩個概念,對寫代碼太重要了:
1、父節(jié)點和子節(jié)點的關(guān)系:看定義
2、完全二叉樹:參考[2]
基本操作
1、Build(構(gòu)建堆)
2、Insert(插入)
3、Delete(刪除:最小或者最大的那個)
代碼實現(xiàn)
首先,寫代碼前有兩個非常重要的點:
1、用一個數(shù)組就可以作為堆的存儲結(jié)構(gòu),非常簡單而且易操作;
2、另外同樣因為是數(shù)組作為存儲結(jié)構(gòu),所以父子節(jié)點之間的關(guān)系就能根據(jù)索引就輕松找到對方了。
對于JavaScript以0作為數(shù)組索引開始,關(guān)系如下:
nLeftIndex = 2 * (nFatherIndex+1) - 1; nRightIndex = 2* (nFatherIndex+1);
前面提到注意兩個概念,是有助于理解的:
1、因為是數(shù)組,所以父子節(jié)點的關(guān)系就不需要特殊的結(jié)構(gòu)去維護了,索引之間通過計算就可以得到,省掉了很多麻煩。如果是鏈表結(jié)構(gòu),就會復(fù)雜很多;
2、完全二叉樹的概念可以參考[2],要求葉子節(jié)點從左往右填滿,才能開始填充下一層,這就保證了不需要對數(shù)組整體進(jìn)行大片的移動。這也是隨機存儲結(jié)構(gòu)(數(shù)組)的短板:刪除一個元素之后,整體往前移是比較費時的。這個特性也導(dǎo)致堆在刪除元素的時候,要把最后一個葉子節(jié)點補充到樹根節(jié)點的緣由
代碼實現(xiàn):
/****************************************************** * file : 堆 * author : "page" * time : "2016/11/02" *******************************************************/ function Heap() { this.data = []; } Heap.prototype.print = function () { console.log("Heap: " + this.data); } Heap.prototype.build = function(data){ // 初始化 this.data = []; if (!data instanceof Array) return false; // 入堆 for (var i = 0; i < data.length; ++i) { this.insert(data[i]); } return true; } Heap.prototype.insert = function( nValue ){ if (!this.data instanceof Array) { this.data = []; } this.data.push(nValue); // 更新新節(jié)點 var nIndex = this.data.length-1; var nFatherIndex = Math.floor((nIndex-1)/2); while (nFatherIndex > 0){ if (this.data[nIndex] < this.data[nFatherIndex]) { var temp = this.data[nIndex]; this.data[nIndex] = this.data[nFatherIndex]; this.data[nFatherIndex] = temp; } nIndex = nFatherIndex; nFatherIndex = Math.floor((nIndex-1)/2); } } Heap.prototype.delete = function( ){ if (!this.data instanceof Array) { return null; } var nIndex = 0; var nValue = this.data[nIndex]; var nMaxIndex = this.data.length-1; // 更新新節(jié)點 var nLeaf = this.data.pop(); this.data[nIndex] = nLeaf; while (nIndex < nMaxIndex ){ var nLeftIndex = 2 * (nIndex+1) - 1; var nRightIndex = 2 * (nIndex+1); // 找最小的一個子節(jié)點(nLeftIndex < nRightIndex) var nSelectIndex = nLeftIndex; if (nRightIndex < nMaxIndex) { nSelectIndex = (this.data[nLeftIndex] > this.data[nRightIndex]) ? nRightIndex : nLeftIndex; } if (nSelectIndex < nMaxIndex && this.data[nIndex] > this.data[nSelectIndex] ){ var temp = this.data[nIndex]; this.data[nIndex] = this.data[nSelectIndex]; this.data[nSelectIndex] = temp; } nIndex = nSelectIndex; } return nValue; } // test var heap = new Heap(); heap.build([1, 3, 5, 11, 4, 6, 7, 12, 15, 10, 9, 8]); heap.print(); // insert heap.insert(2); heap.print(); // delete heap.delete(); heap.print();
關(guān)于JavaScript的幾點小結(jié)
這里是采用面向?qū)ο蟮囊环N實現(xiàn)方法,感覺上不是太優(yōu)雅,不知道還有沒有更好的表示方法和寫法;
學(xué)習(xí)了數(shù)組的幾個用法:push和pop的操作太好用了;
判斷數(shù)組的方式也是臨時從網(wǎng)上搜的(instanceof),印象不深刻,不用的話下次估計還是有可能忘掉。
參考
[1]《數(shù)據(jù)結(jié)構(gòu)和算法分析:C語言描述》
[2]圖解數(shù)據(jù)結(jié)構(gòu)(8)——二叉堆
總結(jié)
JavaScript的數(shù)組實現(xiàn)了push和pop這些操作,許多其他語言也提供了類似的數(shù)據(jù)結(jié)構(gòu)和操作(比如C++的Vector),同時也支持隨機操作。所以,我開始想如果這些結(jié)構(gòu)上簡單的加上自動排序的概念,那么一個堆就輕松搞定了,后面看到C++ STL的make_heap就知道自己知道的太少了,但也慶幸自己思維方式是對的。JavaScript的沒有去查,我想有或者實現(xiàn)起來很容易;
自己去實現(xiàn)了之后,發(fā)現(xiàn)這個結(jié)構(gòu)也很簡單,只要你肯去跟它親密接觸一次就可以了;
JavaScript的細(xì)節(jié)部分還是不太了解,比如數(shù)組的應(yīng)用上還要再翻資料才能用;對于JavaScript的靈魂還是沒有接觸到,精髓部分需要不斷的學(xué)習(xí)和練習(xí);
這些代碼,只要你去了解了概念,了解了編程的基礎(chǔ),就可以寫的出來。但是,代碼還可以寫的更簡潔,比如delete函數(shù)求最小的子節(jié)點的時候,左右節(jié)點的索引就不需要比較,肯定是左邊的小。代碼部分感覺還是可以繼續(xù)優(yōu)化和精簡的。
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
- JavaScript調(diào)用堆棧及setTimeout使用方法深入剖析
- java自帶的工具Jstack截取進(jìn)程中的堆棧信息
- JS實現(xiàn)隊列與堆棧的方法
- js自動生成的元素與頁面原有元素發(fā)生堆疊的解決方法
- Javascript堆排序算法詳解
- 使用堆實現(xiàn)Top K算法(JS實現(xiàn))
- JavaScript實現(xiàn)顯示函數(shù)調(diào)用堆棧的方法
- js自動生成的元素與頁面原有元素發(fā)生堆疊的解決方法
- JavaScript把數(shù)組作為堆棧使用的方法
- JavaScript數(shù)組實現(xiàn)數(shù)據(jù)結(jié)構(gòu)中的隊列與堆棧
相關(guān)文章
javascript獲取以及設(shè)置光標(biāo)位置
本文介紹了javascript獲取以及設(shè)置光標(biāo)位置的方法,具有很好的參考價值,下面跟著小編一起來看下吧2017-02-02微信小程序?qū)崿F(xiàn)側(cè)邊導(dǎo)航欄
這篇文章主要為大家詳細(xì)介紹了微信小程序?qū)崿F(xiàn)側(cè)邊導(dǎo)航欄,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-07-07javascript+jQuery實現(xiàn)360開機時間顯示效果
這篇文章主要介紹了javascript+jQuery實現(xiàn)360開機時間顯示效果,在文中給大家提到了js實現(xiàn)時間倒計時的代碼,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2017-11-11JavaScript實現(xiàn)的內(nèi)存數(shù)據(jù)庫LokiJS介紹和入門實例
這篇文章主要介紹了JavaScript實現(xiàn)的內(nèi)存數(shù)據(jù)庫LokiJS介紹和入門實例,LokiJS是一個內(nèi)存數(shù)據(jù)庫,將性能考慮放在第一位,使用JavaScript編寫,需要的朋友可以參考下2014-11-11