成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

數(shù)據(jù)結(jié)構(gòu)-棧

zhoutk / 2039人閱讀

摘要:棧是另外一種數(shù)據(jù)結(jié)構(gòu),類似于數(shù)組,但是在添加或刪除數(shù)據(jù)時更加靈活。棧數(shù)據(jù)結(jié)構(gòu)棧是一種后進先出的數(shù)據(jù)結(jié)構(gòu)。這種情況下,可以直接通過修改來修改棧中的數(shù)據(jù),這是無法避免的。

前言

數(shù)組是 JS 中最常用的數(shù)據(jù)結(jié)構(gòu),它可以在任意位置添加或刪除數(shù)據(jù)。棧是另外一種數(shù)據(jù)結(jié)構(gòu),類似于數(shù)組,但是在添加或刪除數(shù)據(jù)時更加靈活。

棧數(shù)據(jù)結(jié)構(gòu)

棧是一種 后進先出(LIFO) 的數(shù)據(jù)結(jié)構(gòu)。新添加或待刪除的元素都保存在棧的一端,叫 棧頂 ,另一端就叫做 棧底 。在棧中,新元素都靠近棧頂,就元素都靠近棧底。

創(chuàng)建棧

可以用數(shù)組來模擬一個棧結(jié)構(gòu):

function Stack() {
    let items = []
    // 棧的屬性和方法
}

需要實現(xiàn)的方法:

push(element): 添加一個元素到棧頂

pop(): 移除棧頂?shù)脑?,并返回該元?/p>

peek(): 僅僅返回棧頂?shù)脑?/p>

clear(): 清空棧

size(): 返回棧中的元素的個數(shù)

isEmpty(): 判斷棧是否為空

// 向棧中添加元素
this.push = function (element) {
    items.push(element)
}
// 從棧中移除元素
this.pop = function () {
    return items.pop()
}
// 查看棧頂元素
this.peek = function () {
    return items[item.length - 1]
}
// 檢查棧是否為空
this.isEmpty = function () {
    return !!item.length
}
// 清空棧中的元素
this.clear = function () {
    items = []
}
// 返回棧的大小
this.size = function () {
    return items.length
}
// 打印棧
this.print = function () {
    console.log(items.toString())
}
ES6 與 棧

ES6 的寫法:

class Stack {
    constructor() {
        this.items = []
    }
    push (element) {
        this.items.push(element)
    }
    // ... 其他方法
}

ES6 的類是基于原型的,雖然基于原型的類比基于函數(shù)的類更節(jié)省內(nèi)存,但是卻不能聲明私有變量,所以變量 items 是公共的。這種情況下,可以直接通過修改 items 來修改棧中的數(shù)據(jù),這是無法避免的。

用 ES6 的限定作用域 Symbol 實現(xiàn)類

ES6 新增了 Symbol 基礎(chǔ)類型,它是不可變的,也可以作用對象的屬性。

let _items = Symbol()
class Stack {
    constructor() {
        this[_items] = []
    }
    // ... 其他方法
}

上面這個例子創(chuàng)建了一個假的私有屬性,不能完全規(guī)避上面提到的問題,因為 ES6 新增的 Object.getOwnPropertySymbols 方法能夠取到類里面聲明的所有 Symbols 屬性,比如:

let stack = new Stack()
stack.push(66)
stack.push(88)
let objectSymbols = Object.getOwnPropertySymbols(stack)
console.log(objectSymbols.length) // 1
console.log(objectSymbols[0]) // Symbol()
stack[objectSymbols[0]].push(1)
stack.print() // 66 88 1
通過訪問 stack[objectSymbols[0]] 是可以訪問 _items 的,并且可以對 _items 進行任意操作。
用 ES6 的WeakMap 實現(xiàn)類

有一種數(shù)據(jù)類型可以確保屬性是私有的,這就是 WeakMap 。WeakMap 可以存儲鍵值對,其中鍵是對象,值可以是任意數(shù)據(jù)類型。

const items = new WeakMap()
class Stack {
    constructor() {
        items.set(this, [])
    }
    push(element) {
        let s = items.get(this)
        s.push(element)
    }
    pop() {
        let s = items.get(this)
        return s.pop()
    }
    // ... 其他方法
}

現(xiàn)在,Stack 中的 items 是私有的了,但是 items 是在 Stack 類以外聲明的,還是可以被改動,所以需要借助閉包來實現(xiàn)一層封裝:

let Stack = (function () {
    const items = new WeakMap()
    class Stack {
    constructor() {
        items.set(this, [])
    }
    // ... 其他方法
    return Stack
}
})()

### 用棧解決實際問題
棧在 JS 中應(yīng)用還是十分廣泛的,比如 調(diào)用棧 。進制轉(zhuǎn)換也是很常見的例子,可以用 棧 來處理,比如要把十進制轉(zhuǎn)化成二進制,可以將該十進制數(shù)字和2整除,直到結(jié)果是 0 為止。

function divideBy2 (decNumber) {
    var remStack = new Stack(),
        rem,
        binaryString = ""
   
   while (decNumber > 0) {
       rem = Math.floor(decNumber % 2)
       remStack.push(rem)
       decNumber = Math.floor(decNumber / 2)
   }
   while (!remStack.isEmpty()) {
       binaryString += remStack.pop().toString()
   }
   return binaryString
}

這個例子中,當(dāng)結(jié)果滿足和2做整除的條件是,會取得當(dāng)前結(jié)果和2的余數(shù),放到棧里,然后讓結(jié)果繼續(xù)和2做整除。

#### 改進算法
把上面的例子改成十進制轉(zhuǎn)成任意進制的:

function baseConverter(decNumber, base) {
     var remStack = new Stack(),
        rem,
        binaryString = "",
        digits = "0123456789ABCDEF"
   
     while (decNumber > 0) {
        rem = Math.floor(decNumber % base)
        remStack.push(rem)
        decNumber = Math.floor(decNumber / base)
     }
     while (!remStack.isEmpty()) {
        binaryString += digits[remStack.pop()]
     }
   return binaryString
}

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/101448.html

相關(guān)文章

  • js數(shù)據(jù)結(jié)構(gòu)和算法(二)和隊列

    摘要:對于棧來說,這個表尾稱為棧的棧頂,相應(yīng)的表頭稱為棧底。棧和隊列的區(qū)別棧的插入和刪除操作都是在一端進行的,而隊列的操作卻是在兩端進行的。出棧操作出棧操作就是在棧頂取出數(shù)據(jù),棧頂指針隨之下移的操作。 基本概念 棧和隊列都是動態(tài)的集合,在棧中,可以去掉的元素是最近插入的哪一個。棧實現(xiàn)了后進先出。在隊列中,可以去掉的元素總是在集合中存在的時間最長的那一個。隊列實現(xiàn)了先進先出的策略。 棧的官...

    jsummer 評論0 收藏0
  • Java版-數(shù)據(jù)結(jié)構(gòu)-

    摘要:介紹棧是一種后進先出的線性表數(shù)據(jù)結(jié)構(gòu),分為棧頂和棧底兩端,僅允許在表的一端插入元素,這一端被稱為棧頂,另外一端稱之為棧底。 介紹 棧是一種后進先出的線性表數(shù)據(jù)結(jié)構(gòu),分為棧頂和棧底兩端,僅允許在表的一端插入元素,這一端被稱為棧頂,另外一端稱之為棧底。棧,只有兩種操作,分為入棧(壓棧)和出棧(退棧);向棧中添加元素的操作叫做入棧,相反從棧中刪除元素叫做出棧。 特點 只能從棧頂添加元素或者...

    voidking 評論0 收藏0
  • JavaScript數(shù)據(jù)結(jié)構(gòu)

    摘要:我們都知道數(shù)組是里面比較常用的一種數(shù)據(jù)結(jié)構(gòu),棧和數(shù)組類似,定義如下棧是一種遵從后進先出原則的有序集合。新增加和待刪除的元素都保存在棧的尾部,也稱棧頂,相反的另一端就叫棧底,在棧的這種數(shù)據(jù)結(jié)構(gòu)里面,我們新增的元素都在棧頂,舊的元素都在棧底。 由于不是計算機專業(yè)出身,對數(shù)據(jù)結(jié)構(gòu)這些了解的比較少,最近看了一些相關(guān)的書籍,這里做一些總結(jié)。本篇要說的是棧。我們都知道數(shù)組是JavaScript里面...

    nanchen2251 評論0 收藏0
  • finally與return之間的關(guān)系

    摘要:則會在轉(zhuǎn)移指令前執(zhí)行??偨Y(jié)與之間的關(guān)系如果在中含有語句,那么語句的還有作用嗎先看一段代碼如果你對內(nèi)存布局不是很清楚,請看這篇文章虛擬機類加載機制和字節(jié)碼執(zhí)行引擎重點關(guān)注運行時棧幀結(jié)構(gòu)局部變量表槽,操作數(shù)棧。 定論 問:finally語句一定會執(zhí)行嗎?答: 如果沒有執(zhí)行相應(yīng)的try語句則不會執(zhí)行。 在try語句中如果調(diào)用System.exit(0)方法則不會執(zhí)行。 問:finally...

    Yuanf 評論0 收藏0
  • JS數(shù)據(jù)結(jié)構(gòu)學(xué)習(xí):

    摘要:棧的應(yīng)用前面介紹了那么多棧相關(guān)的知識,最后也是介紹棧的應(yīng)用場景的時候了,棧的實際應(yīng)用非常廣泛,例如用來存儲訪問過的任務(wù)或路徑撤銷的操作。 棧的定義 什么是棧?棧是一種遵循后進先出原則的有序集合,新添加的或者待刪除的元素都保存在棧的同一端,稱為棧頂,另一端稱為棧底,在棧里,新元素靠近棧頂,舊元素靠近棧底,用個圖來看大概這樣式的:showImg(https://segmentfault.c...

    Alfred 評論0 收藏0
  • js 調(diào)用機制與ES6尾調(diào)用優(yōu)化介紹

    摘要:調(diào)用棧的運行機制機制程序運行到一個函數(shù),它就會將其添加到調(diào)用棧中,當(dāng)從這個函數(shù)返回的時候,就會將這個函數(shù)從調(diào)用棧中刪掉。在調(diào)用棧中每個調(diào)用偵都對應(yīng)一個函數(shù),最上方的調(diào)用幀稱為當(dāng)前幀,調(diào)用棧是由所有的調(diào)用偵形成的。 showImg(https://segmentfault.com/img/remote/1460000019244497?w=900&h=600); 調(diào)用棧的英文名叫做Cal...

    AaronYuan 評論0 收藏0

發(fā)表評論

0條評論

zhoutk

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<