热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

数据结构详解:链栈中的元素入栈与出栈操作分析

链栈虽然通常以数组作为底层实现,但也可以采用链表来构建Stack类。在这种情况下,空堆栈通过NULL指针表示。当新元素被压入堆栈时,它会被添加到链表的头部,从而实现高效的入栈操作。此外,出栈操作则通过移除链表头部的节点来完成,确保了操作的时间复杂度为O(1)。这种设计不仅简化了内存管理,还提高了动态数据处理的灵活性。

链栈

尽管数组是堆栈最常见的底层表示形式,但也可以使用链表实现Stack类。 如果这样做,空堆栈的概念表示就是NULL指针:
这里写图片描述
当你将新元素推入堆栈时,该元素只会添加到链表的前端。因此,如果将元素e1 push到空栈上,该元素将存储在一个新单元格中,该单元格将成为链中唯一的链接:
这里写图片描述
将一个新元素push栈将该元素添加到链的开头。所涉及的步骤与将字符插入链表列表缓冲区所需的步骤相同。 首先分配一个新的单元格,然后输入数据,最后更新链接指针,使新单元格成为链中的第一个元素。 因此,如果将元素e2推到栈上,将得到以下配置:
这里写图片描述
在链表表示中,pop操作包括删除链中的第一个单元格并返回存储在该列中的值。因此,来自上图所示的栈的pop操作返回e2并恢复栈的先前状态,如下所示:
这里写图片描述

对链栈中pop和push的分析

先看一下这两个功能的实现(我们以数字栈为例子):

class StackInt {          // in StackInt.h
public:
StackInt (); // constructor

void push(int value); // append a value
int pop(); // return the first-in value

private:
struct Node {
int value;
Node * link;
};
Node * data; // member variables
};

我们先看看我们的push功能:

void StackInt::push(int v) {
Node * temp = new Node;
temp->value = v;
temp->link = data;
data = temp;
}

链栈是链表的一种,我们当然要画图去理解这个过程:
(1) Node* data这句话的意思我们说过,就是建立一个指向这组数据的首指针,如图:
这里写图片描述
(2)这个时候我们要push(7),因此我们的目的应该是这样的:
这里写图片描述
(3)代码的前两句,是建立一个指向新的Node的指针,然后为新建的节点赋值为7,如下:
这里写图片描述
(4)这里我们好像可以把第三句删了,直接 data = temp; 让我们看看会发生什么?
这里写图片描述
(5)这里我们很明显看到,如果这样做,那么data就直接指向新的节点,但是新的节点并没有链接到我们的下一个数据,相当于断开了,因此正确的应该是这样:
这里写图片描述
然后:
这里写图片描述
(6)最后我们可以执行删除操作,把我们的temp空间删除,大功告成!!
这里写图片描述

我们再看看我们的pop功能

int StackInt::pop() {
int toReturn = data->value;
Node * temp = data;
data = temp->link;
delete temp;
return toReturn;
}

(1)首先我们把我们要pop出去的值给保存起来,定义一个新的变量:
这里写图片描述
(2)如果我们直接用 data = data->link;而不建立临时的空间,我们来看看如果这样会怎么样
这里写图片描述
(3)这显然不是我们想要的,所以我们接着再试一次,首先我们建立临时变量:
这里写图片描述
(4)然后我们将temp的link直接赋值给我们的data。如下;
这里写图片描述
(5)最后删除节点,然后返回原来的值就ok了
这里写图片描述

push和pop的算法复杂度

由于我们并没有涉及到其他的数据移动,我们只是进行了一次的插入删除操作,所以他们的算法复杂度均为我们的O(1)

下一篇我们就用通用的模板来实现我们一般的链栈结构!


推荐阅读
  • Codeforces Round #566 (Div. 2) A~F个人题解
    Dashboard-CodeforcesRound#566(Div.2)-CodeforcesA.FillingShapes题意:给你一个的表格,你 ... [详细]
  • 题目Link题目学习link1题目学习link2题目学习link3%%%受益匪浅!-----&# ... [详细]
  • 火星商店问题:线段树分治与持久化Trie树的应用
    本题涉及编号为1至n的火星商店,每个商店有一个永久商品价值v。操作包括每天在指定商店增加一个新商品,以及查询某段时间内某些商店中所有商品(含永久商品)与给定密码值的最大异或结果。通过线段树分治和持久化Trie树来高效解决此问题。 ... [详细]
  • 题目描述:给定n个半开区间[a, b),要求使用两个互不重叠的记录器,求最多可以记录多少个区间。解决方案采用贪心算法,通过排序和遍历实现最优解。 ... [详细]
  • 在金融和会计领域,准确无误地填写票据和结算凭证至关重要。这些文件不仅是支付结算和现金收付的重要依据,还直接关系到交易的安全性和准确性。本文介绍了一种使用C语言实现小写金额转换为大写金额的方法,确保数据的标准化和规范化。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 深入理解C++中的KMP算法:高效字符串匹配的利器
    本文详细介绍C++中实现KMP算法的方法,探讨其在字符串匹配问题上的优势。通过对比暴力匹配(BF)算法,展示KMP算法如何利用前缀表优化匹配过程,显著提升效率。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 本文详细介绍了如何构建一个高效的UI管理系统,集中处理UI页面的打开、关闭、层级管理和页面跳转等问题。通过UIManager统一管理外部切换逻辑,实现功能逻辑分散化和代码复用,支持多人协作开发。 ... [详细]
  • 本文探讨了如何在模运算下高效计算组合数C(n, m),并详细介绍了乘法逆元的应用。通过扩展欧几里得算法求解乘法逆元,从而实现除法取余的计算。 ... [详细]
  • 扫描线三巨头 hdu1928hdu 1255  hdu 1542 [POJ 1151]
    学习链接:http:blog.csdn.netlwt36articledetails48908031学习扫描线主要学习的是一种扫描的思想,后期可以求解很 ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 本文探讨了 C++ 中普通数组和标准库类型 vector 的初始化方法。普通数组具有固定长度,而 vector 是一种可扩展的容器,允许动态调整大小。文章详细介绍了不同初始化方式及其应用场景,并提供了代码示例以加深理解。 ... [详细]
  • 本文详细介绍了 React 中的两个重要 Hook 函数:useState 和 useEffect。通过具体示例,解释了如何使用它们来管理组件状态和处理副作用。 ... [详细]
author-avatar
拍友2602939213
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有