【JavaScript数据结构系列】02-栈Stack

码路工人 CoderMonkey

转载请注明作者与出处

## 1. 认识栈结构

栈是非常常用的一种数据结构,
与数组同属线性数据结构,
不同于数组的是它是一种受限的线性结构。

画一张图来说明:




如上图所示,

  • 最新入栈的元素/最底下的元素,为栈底
  • 最后一个/最上面的元素,为栈顶
  • 最后一个入栈元素最先出栈(LIFO原则)
  • 只能操作栈顶
  • 添加元素叫:进栈/压栈/入栈
  • 删除元素叫:出栈/退栈

## 2. 栈的应用:

  • 函数调用栈
  • 文本编辑器中的撤销与重做

等。

3. 栈的实现

注:

ES6 版的代码实现请查看 npm 包 data-struct-js 代码

代码在 Github/Gitee

3.1 常用方法

方法 描述
push(element) 添加元素到栈顶
pop() 删除栈顶元素
peek() 查看栈顶元素
isEmpty() 检查是否为空栈
size() 检查栈容量
clear() 清空栈
toString() 字符串化

3.2 常用方法的代码实现

栈的实现可以基于数组,也可以基于链表,
这里我们用基于数组来实现一下。

首先,写出Stack的构造函数

function Stack() {
this.__items = []
}

3.2.1 push

实现分析:

向数组尾部添加元素,同JS数组操作

function Stack() {
this.__items = [] Stack.prototype.push = function(element) {
return this.__items.push(element)
}
}

3.2.2 pop

实现分析:

删除/弹出数组最后一个元素,同JS数组操作

function Stack() {
this.__items = [] Stack.prototype.pop = function() {
return this.__items.pop()
}
}

3.2.3 peek

实现分析:

查看栈顶即数组最后一个元素

function Stack() {
this.__items = [] Stack.prototype.peek = function() {
if(!this.__items.length) return null
return this.__items[this.__items.length-1]
}
}

3.2.4 isEmpty

实现分析:

只要看内部数组元素个数是否为0

function Stack() {
this.__items = [] Stack.prototype.isEmpty = function() {
return this.__items.length === 0
}
}

3.2.5 size

实现分析:

返回内部数组元素个数

function Stack() {
this.__items = [] Stack.prototype.size = function() {
return this.__items.length
}
}

3.2.6 clear

实现分析:

清空内部数组

function Stack() {
this.__items = [] Stack.prototype.clear = function() {
this.__items.length = 0
}
}

3.2.7 toString

实现分析:

将内部数组用 join 连结

function Stack() {
this.__items = [] Stack.prototype.toString = function() {
return this.__items.join(' ')
}
}

完整代码如下:

// 栈
function Stack() {
this.__items = []
// 入栈
Stack.prototype.push = function (element) {
return this.__items.push(element)
}
// 出栈
Stack.prototype.pop = function () {
return this.__items.pop()
}
// 查看栈顶
Stack.prototype.peek = function () {
if (!this.__items.length) return null
return this.__items[this.__items.length - 1]
}
// 是否为空栈
Stack.prototype.isEmpty = function () {
return this.__items.length === 0
}
// 获取栈大小/容量/元素个数​
Stack.prototype.size = function () {
return this.__items.length
}
// 清空栈
Stack.prototype.clear = function() {
this.__items.length = 0
}
// 字符串值
Stack.prototype.toString = function () {
return this.__items.join(' ')
}
}

3.3 下面我们测试一下:

// ---------------------------------------------
// Test
// ---------------------------------------------
var s = new Stack()
for(var i = 0; i < 5; i++){
s.push(i)
}
console.log('isEmpty: ',s.isEmpty()) // isEmpty: false
console.log('size: ',s.size()) // size: 5
console.log(s.toString()) // 0 1 2 3 4
while(s.size()) {
console.log(`pop: `, s.pop()) // 4 3 2 1 0
}
console.log('isEmpty: ',s.isEmpty()) // isEmpty: true
console.log('size: ',s.size()) // size: 0

我们得到了符合预期的结果。

在上面的实现中,没有考虑复杂类型时的引用传递问题,

也没有遍历方法,这些我们将在后面补充完善。

4. 思考题

4.1 判断哪个不是可能的出栈顺序?()

有六个元素`6,5,4,3,2,1`依次进栈,下列哪一个不是合法的出栈顺序?
- A. 5 4 3 6 1 2
- B. 4 5 3 2 1 6
- C. 3 4 6 5 2 1
- D. 2 3 4 1 5 6

这个选择题还是非常简单的,
满足先进后出画图一试答案就出来了。


下面利用栈结构实现十进制转二进制。
比起网上的其它例子,
这里稍微复杂了的一点在于,
不仅仅针对正整数,考虑了负数。

4.2 十进制整数转二进制的方法实现

4.2.1 转换的数学方法

将一个十进制的数转为二进制,

  • 将这个数与2取模,记录余数,将商用于下一次计算
  • 重复上一步,直到商为0
  • 将得到的余数由后向前连接起来,即为所求二进制结果

举例:计算8的2进制值是多少

原十进制值 取模 等于 余数
8 2 4 0
4 2 2 0
2 2 1 0
1 2 0 1

这样,得到的二进制就是:1000
(也即 0000 1000)

负数时的规则:反码后加1补码
所以 -8 的二进制计算过程为:

  • 0000 1000取反:1111 0111
  • 补码:1111 1000

4.2.2 转换的代码实现

基于栈的实现,这里就不重复贴 Stack.js 的代码了。

代码实现分析:

  • 等于0的情况,直接返回0
  • 大于0的情况,按照我们上面列出的转换方法

    • 与2取模,保存入栈,直到商为0
    • 依序出栈,拼接结果并返回
  • 小于0的情况,比起以上两种情况来说最复杂

    • 先乘以-1转为正整数,再按照大于0的情况进行取模处理
    • 取模结果要取反(即0变1,1变0),然后保存入栈
    • 计算完,依序出栈,得到中间结果
    • 将上面得到的结果值 +1 进行补码
    • 再在首位前加上一位1来表示负数
    • TODO:其实还需要完善一点,空位用1补足8位(或其它可能的位数)
// ---------------------------------------------
// decimal to binary
// ---------------------------------------------
function dec2bin(decNum) {
if(!decNum) return 0 var stack = new Stack()
var minus = decNum < 0 if(minus) {
decNum *= -1
while(decNum) {
var temp = decNum % 2
switch(temp) {
case 0:
temp = 1
break
case 1:
temp = 0
break
}
stack.push(temp)
decNum = parseInt(decNum / 2)
}
} else {
while(decNum) {
stack.push(decNum % 2)
decNum = parseInt(decNum / 2)
}
} var result = ''
while(!stack.isEmpty()) {
result += stack.pop()
} if(minus) {
// 补码
for(var i=result.length-1;i>=0;i--) {
var arrTemp = result.split('')
switch(result[i]) {
case '0':
arrTemp[i] = 1
break
case '1':
arrTemp[i] = 0
break
}
result = arrTemp.join('')
}
result = '1' + result
} return result
} var ret = dec2bin(8)
console.log(' 8 => ', ret) // 1000 ret = dec2bin(-8)
console.log('-8 => ', ret) // 1 1000

以上就是完整代码,运行一下试试吧。


基于 ES6 实现的 JavaScript 数据结构,
虽然这个小轮子很少会被使用,
也许对于初学者学习 JavaScript 会有点帮助。
只要简单 install 一下即可,感兴趣的话还可以去
GitHub / Gitee 看源码。(Star 表支持~)

npm install data-struct-js --save-dev

https://github.com/CoderMonkie/data-struct-js

https://gitee.com/coder-monkey/data-struct-js

最后,感谢您的阅读和支持~


-end-

【JavaScript数据结构系列】02-栈Stack的更多相关文章

  1. 【JavaScript数据结构系列】03-队列Queue

    [JavaScript数据结构系列]03-队列Queue 码路工人 CoderMonkey 转载请注明作者与出处 1. 认识队列Queue结构 队列,跟我们的日常生活非常贴近,我们前面举例了食堂排队打 ...

  2. 【JavaScript数据结构系列】00-开篇

    [JavaScript数据结构系列]00-开篇 码路工人 CoderMonkey 转载请注明作者与出处 ## 0. 开篇[JavaScript数据结构与算法] 大的计划,写以下两部分: 1[JavaS ...

  3. javascript数据结构与算法---栈

    javascript数据结构与算法---栈 在上一遍博客介绍了下列表,列表是最简单的一种结构,但是如果要处理一些比较复杂的结构,列表显得太简陋了,所以我们需要某种和列表类似但是更复杂的数据结构---栈 ...

  4. JavaScript数据结构与算法-栈练习

    栈的实现 // 栈类 function Stack () { this.dataStore = []; this.top = 0; // 栈顶位置 相当于length,不是索引. this.push ...

  5. 【JavaScript数据结构系列】04-优先队列PriorityQueue

    [JavaScript数据结构系列]04-优先队列PriorityQueue 码路工人 CoderMonkey 转载请注明作者与出处 ## 1. 认识优先级队列 经典的案例场景: 登机时经济舱的普通队 ...

  6. 【JavaScript数据结构系列】01-数组Array

    [JavaScript数据结构系列]01-数组Array 码路工人 CoderMonkey 转载请注明作者与出处 # [JavaScript数据结构系列] # 01-数组Array 数组: 是有序的元 ...

  7. JavaScript进阶系列02,函数作为参数以及在数组中的应用

    有时候,把函数作为参数可以让代码更简洁. var calculator = { calculate: function(x, y, fn) { return fn(x, y); } }; var su ...

  8. 【JavaScript数据结构系列】07-循环链表CircleLinkedList

    [JavaScript数据结构系列]07-循环链表CircleLinkedList 码路工人 CoderMonkey 转载请注明作者与出处 1. 认识循环链表 首节点与尾节点相连的,就构成循环链表.其 ...

  9. 【JavaScript数据结构系列】05-链表LinkedList

    [JavaScript数据结构系列]05-链表LinkedList 码路工人 CoderMonkey 转载请注明作者与出处 ## 1. 认识链表结构(单向链表) 链表也是线性结构, 节点相连构成链表 ...

随机推荐

  1. .user.ini 无法修改/删除 怎么办?

    首先 了解chattr命令: Linux chattr命令用于改变文件属性. 这项指令可改变存放在ext2文件系统上的文件或目录属性,这些属性共有以下8种模式: a:让文件或目录仅供附加用途.b:不更 ...

  2. 轻量化模型:MobileNet v2

    MobileNet v2 论文链接:https://arxiv.org/abs/1801.04381 MobileNet v2是对MobileNet v1的改进,也是一个轻量化模型. 关于Mobile ...

  3. Jenkins 构建 Jmeter 项目

    1.启动 Jenkins(windows 版本) 2.新建自由风格的项目 定时任务 构建操作 安装 HTML Publisher 插件 构建后操作 最后保存构建,查看报告

  4. mysql-case..when知识点总结

    case...when..有两种语法: 第一种: case  case_value when when_value  then statement_list [when when_value  the ...

  5. 什么才是Python的高级编程?大牛总结,绝对让你受益匪浅

    很多刚入门或者还在了解的小伙伴们都会遇到迷茫期吧,就是学完这些基础,函数,字典啥的,好像也做不了什么东西,其实你基础学的扎实的话,是能做很多的事的,学完基础也不要迷茫,因为每门语言都是博大精深的,不是 ...

  6. ES[7.6.x]学习笔记(六)分析器

    在前面的章节中,我们给大家介绍了索引中的映射类型,也就是每一个字段都有一个类型,比如:long,text,date等.这和我们的数据库非常的相似,那么它的不同之处是什么呢?对了,就是全文索引,在ES当 ...

  7. Spring官网阅读(三)自动注入

    上篇文章我们已经学习了1.4小结中关于依赖注入跟方法注入的内容.这篇文章我们继续学习这结中的其他内容,顺便解决下我们上篇文章留下来的一个问题-----注入模型. 文章目录 前言: 自动注入: 自动注入 ...

  8. 用最基本的遍历来实现判断字符串 a 是否被包含在字符串 b 中,并返回第一次出现的位置(找不到返回 -1)

    用最基本的遍历来实现判断字符串 a 是否被包含在字符串 b 中,并返回第一次出现的位置(找不到返回 -1) 例子: a='12';b='1234567'; // 返回 0 a='47';b='1234 ...

  9. 一篇文章彻底理解Redis持久化:RDB和AOF

    为什么需要持久化? Redis对数据的操作都是基于内存的,当遇到了进程退出.服务器宕机等意外情况,如果没有持久化机制,那么Redis中的数据将会丢失无法恢复.有了持久化机制,Redis在下次重启时可以 ...

  10. 053.集群管理-Helm部署及使用

    一 Helm概述 1.1 Helm介绍 Helm 是 Kubernetes 的软件包管理工具.包管理器类似 Ubuntu 中使用的apt.Centos中使用的yum 或者Python中的 pip 一样 ...