初级数据结构——栈

news/2024/11/15 10:50:28/

目录

  • 前言
  • 一、栈的基本概念
  • 二、栈的实现方式
  • 三、栈的性能分析
  • 四、栈的应用场景
  • 五、栈的变体
  • 六、出栈入栈的动态图解
  • 七、代码模版
  • 八、总结
  • 结语

前言

数据结构栈(Stack)是一种线性的数据结构,它只允许在序列的一端(称为栈顶)进行插入和删除操作。这种特性使得栈成为许多算法和问题解决中的有力工具。以下是对栈的详细分析:

一、栈的基本概念

1.定义:栈是一种后进先出(LIFO, Last In First Out)的数据结构,即最后插入的元素将是第一个被移除的元素。

2.基本操作
压栈(Push):将元素添加到栈顶。
弹栈(Pop):移除并返回栈顶的元素。
查看栈顶元素(Peek 或 Top):返回栈顶的元素但不移除它。
检查栈是否为空(IsEmpty):判断栈是否为空。

二、栈的实现方式

1.数组实现:
优点:实现简单,访问速度快。
缺点:需要预先分配固定大小的内存空间,可能存在内存浪费或扩容问题。

2.链表实现:
优点:内存使用灵活,不需要预先分配固定大小的内存空间。
缺点:相对数组实现,访问速度可能较慢(因为需要遍历链表)。

三、栈的性能分析

时间复杂度:
压栈(Push):O(1),因为只需要在栈顶添加元素。
弹栈(Pop):O(1),因为只需要移除栈顶元素。
查看栈顶元素(Peek 或 Top):O(1),因为只需要访问栈顶元素。
检查栈是否为空(IsEmpty):O(1),因为只需要判断栈顶指针是否为空。

空间复杂度:
数组实现:O(n),其中n是栈的容量。
链表实现:O(m),其中m是栈中元素的数量(动态分配内存)。

四、栈的应用场景

函数调用和递归:栈用于保存函数调用和递归调用的上下文信息。
表达式求值:栈用于解析和计算后缀表达式(逆波兰表示法)。
深度优先搜索(DFS):在图的遍历中,栈用于实现深度优先搜索。
括号匹配:栈用于检查表达式中的括号是否匹配。
页面访问历史:在浏览器中,栈用于管理页面的访问历史。
撤销操作:在文本编辑器、图像处理软件等中,栈用于实现撤销(undo)操作。
语法分析:在编译器设计中,栈用于语法分析过程中的符号表管理和操作符优先级处理。

五、栈的变体

双栈:使用两个栈来模拟队列等数据结构
栈的栈:栈中的元素本身也是栈,用于实现更复杂的嵌套数据结构
多维栈:将栈扩展到多维空间,用于处理多维数据。

六、出栈入栈的动态图解

入栈:
在这里插入图片描述
出栈
在这里插入图片描述

七、代码模版

顺序栈

#include<iostream>
#include<exception>
using namespace std;template<typename T>//定制栈里面的元素,就像vector一样
class Stack {
private:int size;//既为栈元素个数又为栈顶位置int capacity;T* data;void resize();
public:Stack():data(new T[capacity]),size(0),capacity(10){}//构造函数,申请类型为T容量为capacity的内存空间,相当于数组~Stack();void push(T x);T top() const;T pop();int getSize() const;bool empty() const;
};template<typename T>
void Stack<T>::resize() {//顺序栈扩容int newCapacity = 2 * capacity;T* newData = new T[newCapacity];for (int i = 0; i < size; i++) {newData[i] = data[i];}delete[]data;data = newData;capacity = newCapacity;
}template<typename T>
Stack<T>::~Stack() {delete[]data;
}template<typename T>
void Stack<T>::push(T x) {if (size == capacity) {resize();}data[size++] = x;
}template<typename T>
T Stack<T>::top() const {if (!size) {throw std::underflow_error("Stack is empty");//如果栈为空即为非法状态,抛出异常}return data[size - 1];//注意栈元素序号从零开始
}template<typename T>
T Stack<T>::pop(){if (!size) {throw std::underflow_error("Stack is empty");}return data[--size];
}template<typename T>
int Stack<T>::getSize() const {return size;
}template<typename T>
bool Stack<T>::empty() const {return size == 0 ? 1 : 0;
}int main() {Stack<int> s;for (int i = 0; i < 5; i++) {s.push(i);}int x=s.pop();cout << x << " " << s.top() << endl;cout << s.getSize() << endl;cout << s.empty() << endl;return 0;
}

链式栈

#include<iostream>
#include<stdexcept>
#include<stack>
using namespace std;template<typename T>
class Stack {
private:struct Node{T data;Node* next;Node(T x):data(x),next(NULL){}};Node* head;int size;
public:Stack():size(0),head(NULL){}~Stack();void push(T x);T top() const;T pop();bool empty();int getSize();
};template<typename T>
Stack<T>::~Stack() {while (head) {Node* t = head;head = head->next;delete t;}
}template<typename T>
void Stack<T>::push(T x) {//插入操作用头插法,即头节点为栈顶Node* newNode = new Node(x);newNode->next = head;head = newNode;size++;
}template<typename T>
T Stack<T>::top() const {if (!head) {throw std::underflow_error("Stack is empty!");}return head->data;
}template<typename T>
T Stack<T>::pop() {if (!head) {throw std::underflow_error("Stack is empty!");}T result = head->data;Node* t = head;head = head->next;delete t;size--;return result;
}template<typename T>
int Stack<T>::getSize() {return size;
}template<typename T>
bool Stack<T>::empty() {return size == 0 ? 1 : 0;
}int main() {int n;while (cin >> n) {Stack<int>s;while (n) {s.push(n % 2);n /= 2;}while (!s.empty()) {int x = s.pop();cout << x;}cout << endl;}return 0;
}

八、总结

栈是一种简单而强大的数据结构,它遵循后进先出的原则,并通过数组或链表等数据结构实现。栈在多种算法和场景中都有广泛应用,包括函数调用、递归、表达式求值、深度优先搜索、括号匹配等。理解栈的基本概念和操作对于掌握计算机科学和数据结构的基础知识至关重要。同时,栈的变体也为解决特定问题提供了更多可能性。

结语

下个作品我会更新有关栈的题库,进行栈的实战巩固知识,当然你也可以去力扣等相关刷题网站找题目刷题。
在这里插入图片描述

想看更多内容可以关注我,看我作品,关注我让我们一起学习编程,希望大家能点赞关注支持一下,让我有持续更新的动力,谢谢大家

在这里插入图片描述


http://www.ppmy.cn/news/1547184.html

相关文章

第八章 利用CSS制作导航栏

8.1 水平顶部导航栏 8.1.1 简单水平导航栏的设计与实现 1. 响应式设计 介绍&#xff1a; 响应式设计是指网页能够根据不同设备屏幕尺寸和分辨率自适应调整布局和样式&#xff0c;以确保在手机、平板和桌面等设备上都能提供良好的用户体验。 CSS实现&#xff1a; media (ma…

NoSQL数据库与关系型数据库的主要区别

NoSQL数据库与关系型数据库在多个方面存在显著区别&#xff0c;以下是对这些主要区别的详细描述&#xff1a; 一、数据存储模型 关系型数据库&#xff1a;使用表格形式存储数据&#xff0c;每个表格由行和列组成&#xff0c;行表示记录&#xff0c;列表示字段。数据之间的关系…

PVE纵览-安装系统卡“Loading Driver”的快速解决方案

PVE纵览-安装系统卡“Loading Driver”的快速解决方案 文章目录 PVE纵览-安装系统卡“Loading Driver”的快速解决方案摘要通过引导参数解决PVE安装卡在“Loading Driver”问题官方解决方法 关键字&#xff1a; PVE、 显卡、 Loading、 Driver、 nomodeset 摘要 在虚拟机…

freemarker 读取template.xml ,通过response 输出文件,解决中文乱码问题

采用 try (Writer writer new OutputStreamWriter(os, “UTF-8”)) UTF-8 内容转换 public static void setResponseHeader(HttpServletResponse response, String fileName) {try {// fileName "中文.xls";try {fileName new String(fileName.getBytes(),"…

jupyter+pycharm内部直接运行

第一步&#xff1a;终端使用conda&#xff0c;切换到目标环境&#xff0c;在该虚拟环境下 下载jupyter支持包&#xff08;pip install jupyter&#xff09; 第二步&#xff1a;解决root用户不能直接运行的问题 创建配置文件 jupyter notebook --generate-config 修改…

openwebui二改界面环境搭建

1、下载源码 https://github.com/open-webui/open-webui 2、编译前端 npm i npm run dev 注意版本要求&#xff1a; Python Version: Python 3.11Node.js Version: 20.10 浏览器访问&#xff1a;http://localhost:5173/ 3、编译后端 cd backend conda create --name op…

kafka面试题part-3

6、kafka如何知道哪个消费者消费哪个分区&#xff1f; 生产者把数据发送给各个分区&#xff0c;每个broker节点都有一个coordinator(协调器)&#xff0c;消费者组对分区进行消费&#xff0c;到底哪个消费者消费哪个分区呢&#xff1f;首先groupId对50取模&#xff0c;看最后的结…

掌握ECMAScript模块化:构建高效JavaScript应用

标题&#xff1a;掌握ECMAScript模块化&#xff1a;构建高效JavaScript应用 在现代JavaScript开发中&#xff0c;模块化编程已经成为一个不可或缺的概念。它帮助我们管理和组织代码&#xff0c;提高代码的复用性和可维护性。本文将深入探讨ECMAScript模块化的实现&#xff0c;…