首页 最新 热门 推荐

  • 首页
  • 最新
  • 热门
  • 推荐

JavaScript 中实现常见数据结构:栈、队列与树

  • 25-02-18 14:01
  • 2777
  • 14028
blog.csdn.net

文章目录

  • JavaScript 中实现常见数据结构:栈、队列与树
    • 引言
      • 一、栈(Stack)
      • 二、队列(Queue)
      • 三、树(Tree)
      • 结语与祝福代码

JavaScript 中实现常见数据结构:栈、队列与树

引言

在前端开发中,理解和掌握基础数据结构是提升代码质量、优化算法性能和解决复杂问题的关键。本文将通过JavaScript语言,深入浅出地介绍三种常用的数据结构——栈(Stack)、队列(Queue)和树(Tree),并辅以实例代码帮助读者更好地理解和运用它们。

一、栈(Stack)

栈是一种遵循"后进先出"(Last In First Out, LIFO)原则的线性数据结构。在JavaScript中,我们可以使用数组或类来模拟栈的行为。

用一个序列图(sequence diagram)来表示其“后进先出”(LIFO)的操作过程。入栈(push)、查看栈顶元素(peek)和出栈(pop)操作。

User Stack push(3) (3 added) push(2) (2 added) push(1) (1 added) peek() (returns 1, stack unchanged) pop() (returns 1) (stack now [2, 3]) pop() (returns 2) (stack now [3]) isEmpty() (returns false) pop() (returns 3) (stack now empty) (isEmpty() returns true) User Stack

下面是一个简单的例子:

class Stack {
  constructor() {
    this.items = [];
  }

  // 入栈
  push(element) {
    this.items.push(element);
  }

  // 出栈
  pop() {
    if (this.isEmpty()) {
      return 'Stack is empty';
    }
    return this.items.pop();
  }

  // 查看栈顶元素
  peek() {
    return this.items[this.items.length - 1];
  }

  // 判断栈是否为空
  isEmpty() {
    return this.items.length === 0;
  }

  // 获取栈的大小
  size() {
    return this.items.length;
  }
}

// 示例用法:
const myStack = new Stack();
myStack.push(1);
myStack.push(2);
console.log(myStack.pop()); // 输出: 2
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39

栈作为一种“后进先出”(Last In First Out, LIFO)的数据结构,其操作主要包含两个核心方法:push用于将元素添加到栈顶,pop用于从栈顶移除并返回最后一个添加的元素。在JavaScript中,我们可以利用数组的内置方法来方便地模拟栈的行为,或者创建一个自定义类以更好地体现栈的逻辑和功能。

使用JavaScript数组模拟栈

由于JavaScript数组提供了push和pop方法,它们恰好符合栈的操作要求,因此可以直接用数组实现栈的功能:

// 使用数组模拟栈
let stack = [];

// 入栈操作
stack.push(1);
stack.push(2);
stack.push(3);

console.log(stack); // 输出: [1, 2, 3]

// 出栈操作
let topElement = stack.pop(); // 从栈顶弹出元素,并赋值给topElement
console.log(topElement); // 输出: 3
console.log(stack); // 输出: [1, 2]
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14

创建自定义栈类

尽管可以简单地使用数组,但为了更清晰地表示栈的语义以及便于扩展功能,我们通常会创建一个自定义栈类:

class Stack {
  constructor() {
    this.items = []; // 内部使用数组存储栈中的元素
  }

  // 入栈方法
  push(element) {
    this.items.push(element);
  }

  // 出栈方法
  pop() {
    if (this.isEmpty()) {
      throw new Error('Stack is empty');
    }
    return this.items.pop();
  }

  // 查看栈顶元素,不改变栈状态
  peek() {
    if (this.isEmpty()) {
      throw new Error('Stack is empty');
    }
    return this.items[this.items.length - 1];
  }

  // 判断栈是否为空
  isEmpty() {
    return this.items.length === 0;
  }

  // 获取栈的大小
  size() {
    return this.items.length;
  }
}

// 示例用法:
const myStack = new Stack();
myStack.push('First');
myStack.push('Second');
console.log(myStack.peek()); // 输出: 'Second'
console.log(myStack.pop()); // 输出: 'Second'
console.log(myStack.isEmpty()); // 输出: false
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44

通过这样的自定义栈类,我们不仅能够直观地进行栈操作,还可以增加额外的方法如peek来查看栈顶元素而无需真正移除它,从而满足更多复杂场景的需求。

二、队列(Queue)

队列遵循“先进先出”(First In First Out, FIFO)原则。同样,我们也可以利用数组或者类实现队列功能。

class Queue {
  constructor() {
    this.items = [];
  }

  // 入队
  enqueue(element) {
    this.items.push(element);
  }

  // 出队
  dequeue() {
    if (this.isEmpty()) {
      return 'Queue is empty';
    }
    return this.items.shift();
  }

  // 查看队首元素
  front() {
    return this.items[0];
  }

  // 判断队列是否为空
  isEmpty() {
    return this.items.length === 0;
  }

  // 获取队列大小
  size() {
    return this.items.length;
  }
}

// 示例用法:
const myQueue = new Queue();
myQueue.enqueue('Front');
myQueue.enqueue('Middle');
console.log(myQueue.dequeue()); // 输出: 'Front'
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39

三、树(Tree)

树是一种非线性的数据结构,它由节点(Node)和边组成,每个节点可以有零个或多个子节点。在JavaScript中,我们可以创建一个表示节点的类,并通过引用的方式构建层级关系。

class Node {
  constructor(data) {
    this.data = data;
    this.children = [];
  }

  addChild(childNode) {
    this.children.push(childNode);
  }
}

class Tree {
  constructor(root) {
    this.root = new Node(root);
  }

  traverseDFS(node = this.root, callback) { // 深度优先遍历示例
    callback(node);
    for (let child of node.children) {
      this.traverseDFS(child, callback);
    }
  }

  traverseBFS() { // 广度优先遍历示例,使用队列
    let queue = [this.root];
    while(queue.length > 0) {
      let current = queue.shift();
      console.log(current.data); // 打印节点值
      queue = [...queue, ...current.children];
    }
  }
}

// 示例用法:
const tree = new Tree('Root');
const child1 = new Node('Child1');
const child2 = new Node('Child2');
tree.root.addChild(child1);
tree.root.addChild(child2);
tree.traverseDFS(); // 深度优先遍历
tree.traverseBFS(); // 广度优先遍历
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41

结语与祝福代码

理解并掌握这些基础数据结构就像是拥有了强大的工具箱,使我们在前端编程世界中游刃有余。让我们一起成长,就如同栈中的元素不断积累,如同队列中的任务逐个完成,如同树般枝繁叶茂,向着技术的高峰攀登!

下面是一个结合栈和字符串的小彩蛋,生成一句祝福:

class ReverseWordsInSentence {
  reverseWords(sentence) {
    const stack = new Stack();
    sentence.split(' ').forEach(word => stack.push(word));
    
    let reversedSentence = '';
    while (!stack.isEmpty()) {
      reversedSentence += stack.pop() + ' ';
    }
    return reversedSentence.trim();
  }
}

const revWords = new ReverseWordsInSentence();
console.log(revWords.reverseWords("Happy learning, fellow developers!")); 
// 输出:"developers! fellow learning, Happy"
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

愿每位开发者都能在学习过程中收获满满的快乐!

注:本文转载自blog.csdn.net的极客岛的文章"https://blog.csdn.net/qq_32594913/article/details/136150561"。版权归原作者所有,此博客不拥有其著作权,亦不承担相应法律责任。如有侵权,请联系我们删除。
复制链接
复制链接
相关推荐
发表评论
登录后才能发表评论和回复 注册

/ 登录

评论记录:

未查询到任何数据!
回复评论:

分类栏目

后端 (14832) 前端 (14280) 移动开发 (3760) 编程语言 (3851) Java (3904) Python (3298) 人工智能 (10119) AIGC (2810) 大数据 (3499) 数据库 (3945) 数据结构与算法 (3757) 音视频 (2669) 云原生 (3145) 云平台 (2965) 前沿技术 (2993) 开源 (2160) 小程序 (2860) 运维 (2533) 服务器 (2698) 操作系统 (2325) 硬件开发 (2492) 嵌入式 (2955) 微软技术 (2769) 软件工程 (2056) 测试 (2865) 网络空间安全 (2948) 网络与通信 (2797) 用户体验设计 (2592) 学习和成长 (2593) 搜索 (2744) 开发工具 (7108) 游戏 (2829) HarmonyOS (2935) 区块链 (2782) 数学 (3112) 3C硬件 (2759) 资讯 (2909) Android (4709) iOS (1850) 代码人生 (3043) 阅读 (2841)

热门文章

101
推荐
关于我们 隐私政策 免责声明 联系我们
Copyright © 2020-2025 蚁人论坛 (iYenn.com) All Rights Reserved.
Scroll to Top