当前位置: 首页 > article >正文

栈的深度解析:从基础实现到高级算法应用——C++实现与实战指南

一、栈的核心算法与应用场景

栈的先进后出特性使其在以下算法中表现优异:

  1. 括号匹配:校验表达式合法性。
  2. 表达式求值:中缀转后缀,逆波兰表达式求值。
  3. 深度优先搜索(DFS):模拟递归调用。
  4. 单调栈:解决区间最值问题。
  5. 函数调用栈:模拟程序执行流程。

二、括号匹配算法

1. 问题描述

给定一个包含()[]{}的字符串,判断其是否合法。

2. 实现代码
#include <stack>
#include <string>
#include <unordered_map>bool isValidParentheses(const std::string& s) {std::stack<char> stack;std::unordered_map<char, char> mapping = {{')', '('},{']', '['},{'}', '{'}};for (char ch : s) {if (mapping.count(ch)) { // 右括号if (stack.empty() || stack.top() != mapping[ch]) {return false;}stack.pop();} else { // 左括号stack.push(ch);}}return stack.empty();
}
3. 关键点
  • 使用哈希表存储括号映射关系。
  • 栈为空时遇到右括号直接返回false
  • 最终栈为空才表示匹配成功。

三、表达式求值算法

1. 中缀转后缀(逆波兰表达式)
#include <stack>
#include <string>
#include <vector>
#include <cctype>std::vector<std::string> infixToPostfix(const std::string& expr) {std::vector<std::string> output;std::stack<char> operators;std::unordered_map<char, int> precedence = {{'+', 1}, {'-', 1},{'*', 2}, {'/', 2},{'^', 3}};for (size_t i = 0; i < expr.size(); ++i) {char ch = expr[i];if (isdigit(ch)) { // 数字直接输出std::string num;while (i < expr.size() && isdigit(expr[i])) {num += expr[i++];}output.push_back(num);--i;} else if (ch == '(') { // 左括号入栈operators.push(ch);} else if (ch == ')') { // 右括号弹出至左括号while (!operators.empty() && operators.top() != '(') {output.push_back(std::string(1, operators.top()));operators.pop();}operators.pop(); // 弹出左括号} else { // 运算符while (!operators.empty() && precedence[ch] <= precedence[operators.top()]) {output.push_back(std::string(1, operators.top()));operators.pop();}operators.push(ch);}}// 弹出剩余运算符while (!operators.empty()) {output.push_back(std::string(1, operators.top()));operators.pop();}return output;
}
2. 逆波兰表达式求值
#include <stack>
#include <vector>
#include <string>int evalRPN(const std::vector<std::string>& tokens) {std::stack<int> stack;for (const std::string& token : tokens) {if (token == "+" || token == "-" || token == "*" || token == "/") {int b = stack.top(); stack.pop();int a = stack.top(); stack.pop();if (token == "+") stack.push(a + b);else if (token == "-") stack.push(a - b);else if (token == "*") stack.push(a * b);else stack.push(a / b);} else {stack.push(std::stoi(token));}}return stack.top();
}

四、单调栈算法

1. 问题描述

给定一个数组,找到每个元素的下一个更大元素(Next Greater Element)。

2. 实现代码
#include <stack>
#include <vector>std::vector<int> nextGreaterElements(const std::vector<int>& nums) {std::vector<int> result(nums.size(), -1);std::stack<int> stack;for (int i = 0; i < nums.size(); ++i) {while (!stack.empty() && nums[stack.top()] < nums[i]) {result[stack.top()] = nums[i];stack.pop();}stack.push(i);}return result;
}
3. 关键点
  • 栈中存储数组下标,便于更新结果。
  • 时间复杂度为O(n),空间复杂度为O(n)。

五、深度优先搜索(DFS)与栈

1. 递归DFS
void dfsRecursive(Node* node) {if (!node) return;// 处理当前节点for (auto child : node->children) {dfsRecursive(child);}
}
2. 迭代DFS(使用栈)
void dfsIterative(Node* root) {if (!root) return;std::stack<Node*> stack;stack.push(root);while (!stack.empty()) {Node* curr = stack.top();stack.pop();// 处理当前节点for (auto it = curr->children.rbegin(); it != curr->children.rend(); ++it) {stack.push(*it); // 子节点逆序压栈}}
}

六、函数调用栈模拟

1. 问题描述

模拟函数调用栈的行为,实现一个简单的解释器。

2. 实现代码
#include <stack>
#include <string>
#include <iostream>void executeFunction(const std::string& name) {std::cout << "Entering function: " << name << std::endl;// 模拟函数执行std::cout << "Exiting function: " << name << std::endl;
}void simulateCallStack() {std::stack<std::string> callStack;callStack.push("main");executeFunction(callStack.top());callStack.push("func1");executeFunction(callStack.top());callStack.push("func2");executeFunction(callStack.top());while (!callStack.empty()) {callStack.pop();if (!callStack.empty()) {std::cout << "Returning to function: " << callStack.top() << std::endl;}}
}

七、总结

栈作为一种基础数据结构,在算法设计中具有广泛的应用。通过深入理解栈的特性和应用场景,可以高效解决括号匹配、表达式求值、单调栈、DFS等问题。同时,栈在系统级编程(如调用栈)中也扮演着重要角色。掌握栈的实现和应用,是提升算法能力和编程水平的关键。

相关文章:

栈的深度解析:从基础实现到高级算法应用——C++实现与实战指南

一、栈的核心算法与应用场景 栈的先进后出特性使其在以下算法中表现优异&#xff1a; 括号匹配&#xff1a;校验表达式合法性。表达式求值&#xff1a;中缀转后缀&#xff0c;逆波兰表达式求值。深度优先搜索&#xff08;DFS&#xff09;&#xff1a;模拟递归调用。单调栈&am…...

IDEA集成DeepSeek

引言 随着数据量的爆炸式增长&#xff0c;传统搜索技术已无法满足用户对精准、高效搜索的需求。 DeepSeek作为新一代智能搜索技术&#xff0c;凭借其强大的语义理解与深度学习能力&#xff0c;正在改变搜索领域的游戏规则。 对于 Java 开发者而言&#xff0c;将 DeepSeek 集成…...

Oracle Trace文件突然增长很多的原因分析及解决办法

Oracle Trace文件突然增长很多可能是由多种原因引起的,例如SQL语句的长时间跟踪、错误的跟踪设置、大量的错误和警告信息等。 一、以下是一些解决Trace文件增长过快的方法: 1.清理旧的Trace文件 可以通过以下命令删除超过一定天数的Trace文件,例如删除3天前的Trace文件: …...

leetcode:627. 变更性别(SQL解法)

难度&#xff1a;简单 SQL Schema > Pandas Schema > Salary 表&#xff1a; ----------------------- | Column Name | Type | ----------------------- | id | int | | name | varchar | | sex | ENUM | | salary | int …...

SQLMesh系列教程-3:SQLMesh模型属性详解

SQLMesh 的 MODEL 提供了丰富的属性&#xff0c;用于定义模型的行为、存储、调度、依赖关系等。通过合理配置这些属性&#xff0c;可以构建高效、可维护的数据管道。在 SQLMesh 中&#xff0c;MODEL 是定义数据模型的核心结构&#xff0c;初学SQLMesh&#xff0c;定义模型看到属…...

Java 中的 HashSet 和 HashMap 有什么区别?

一、核心概念与用途 特性HashSetHashMap接口实现实现 Set 接口&#xff08;存储唯一元素&#xff09;实现 Map 接口&#xff08;存储键值对&#xff09;数据存储存储单个对象&#xff08;元素唯一&#xff09;存储键值对&#xff08;键唯一&#xff0c;值可重复&#xff09;典…...

Kubernetes-master 组件

以下是Kubernetes Master Machine的组件。 etcd 它存储集群中每个节点可以使用的配置信息。它是一个高可用性键值存储&#xff0c;可以在多个节点之间分布。只有Kubernetes API服务器可以访问它&#xff0c;因为它可能具有一些敏感信息。这是一个分布式键值存储&#xff0c;所…...

【Leetcode 952】按公因数计算最大组件大小

题干 给定一个由不同正整数的组成的非空数组 nums &#xff0c;考虑下面的图&#xff1a; 有 nums.length 个节点&#xff0c;按从 nums[0] 到 nums[nums.length - 1] 标记&#xff1b;只有当 nums[i] 和 nums[j] 共用一个大于 1 的公因数时&#xff0c;nums[i] 和 nums[j]之…...

js考核第三题

题三&#xff1a;随机点名 要求&#xff1a; 分为上下两个部分&#xff0c;上方为显示区域&#xff0c;下方为控制区域。显示区域显示五十位群成员的学号和姓名&#xff0c;控制区域由开始和结束两个按钮 组成。点击开始按钮&#xff0c;显示区域里的内容开始滚动&#xff0c;…...

【第4章:循环神经网络(RNN)与长短时记忆网络(LSTM)— 4.6 RNN与LSTM的变体与发展趋势】

引言:时间序列的魔法钥匙 在时间的长河中,信息如同涓涓细流,绵延不绝。而如何在这无尽的数据流中捕捉、理解和预测,正是循环神经网络(RNN)及其变体长短时记忆网络(LSTM)所擅长的。今天,我们就来一场深度探索,揭开RNN与LSTM的神秘面纱,看看它们如何在时间序列的海洋…...

简单几个步骤完成 Oracle 到金仓数据库(KingbaseES)的迁移目标

作为国产数据库的领军选手&#xff0c;金仓数据库&#xff08;KingbaseES&#xff09;凭借其成熟的技术架构和广泛的市场覆盖&#xff0c;在国内众多领域中扮演着至关重要的角色。无论是国家电网、金融行业&#xff0c;还是铁路、医疗等关键领域&#xff0c;金仓数据库都以其卓…...

Java和JavaScript当中的json对象和json字符串分别讲解

Java和JavaScript当中的json对象和json字符串分别讲解 一、Java当中的json对象和json字符串 在 Java 中&#xff0c;JSON 对象和 JSON 字符串有不同的表示和操作方式。 1. JSON 对象&#xff1a; 如果你使用的是 org.json 库&#xff0c;创建 JSON 对象的代码如下&#xff1…...

【第11章:生成式AI与创意应用—11.2 音频与音乐生成的探索与实践】

凌晨三点的录音棚里,制作人小林对着空荡荡的混音台抓狂——广告方临时要求将电子舞曲改编成巴洛克风格,还要保留"赛博朋克"元素。当他在AI音乐平台输入"维瓦尔弟遇见霓虹灯"的瞬间,一段融合羽管键琴与合成器的奇妙旋律喷涌而出,这场人与机器的音乐狂想…...

八、SPI读写XT25数据

8.1 SPI 简介 SPI&#xff08;Serial Peripheral Interface&#xff0c;串行外设接口&#xff09;是一种同步串行通信协议&#xff0c;广泛用于嵌入式系统中连接微控制器与外围设备&#xff0c;如传感器、存储器、显示屏等。 主要特点 1. 全双工通信&#xff1a;支持同时发送…...

Visionpro 齿轮测量

效果展示 一、题目要求 求出最大值&#xff0c;最小值&#xff0c;平均值 二、分析 1.首先要进行模板匹配 2.划清匹配范围 3.匹配小三角的模板匹配 4.卡尺 5.用找圆工具 工具 1.CogPMAlignTool 2.CogCaliperTool 3.CogFindCircleTool 4.CogFixtureTool 三、模板匹…...

Ubuntu20.04部署stable-diffusion-webui环境小记

Ubuntu20.04部署stable-diffusion-webui环境小记 文章目录 前言后视镜视角查看安装文档聊聊我踩的那些坑python3.11的安装执行sudo apt update报错显卡驱动内存优化网络问题无法打开系统设置和网络设置查询GPU使用情况 总结 Stable Diffusion web UI A web interface for Stabl…...

索引以及索引底层数据结构

一、什么是索引&#xff1f; 索引&#xff08;index&#xff09;是数据库高效获取数据的数据结构&#xff08;有序&#xff09;。在数据之外&#xff0c;数据库系统还维护着满足特定查找算法的数据结构&#xff08;B树&#xff09;&#xff0c;这些数据结构以某种方式指向真在…...

开业盛典活动策划方案拆解

道叔来给大家详细剖析咱们方案库里刚收录的这份《蜀大侠火锅店武侠风开业盛典活动策划方案》了&#xff0c;保证让你看完直呼过瘾&#xff0c;收获满满&#xff01; 一、主题创意&#xff1a;武侠风&#xff0c;直击人心 首先&#xff0c;咱们得夸一下这活动的主题——“XXX‘…...

API 接口自动化

HTTP协议 - 白月黑羽 HTTP协议简介 如果客户端是浏览器&#xff0c;如何在chrome浏览器中查看 请求和响应的HTTP消息&#xff1f;按f12-》network 清除当前信息 响应的消息体在Response里看 点preview&#xff0c;可以看响应的消息体展开的格式 HTTP请求消息 请求头 reques…...

安全测试|SSRF请求伪造

前言 SSRF漏洞是一种在未能获取服务器权限时&#xff0c;利用服务器漏洞&#xff0c;由攻击者构造请求&#xff0c;服务器端发起请求的安全漏洞&#xff0c;攻击者可以利用该漏洞诱使服务器端应用程序向攻击者选择的任意域发出HTTP请求。 很多Web应用都提供了从其他的服务器上…...

ML.NET库学习008:使用ML.NET进行心脏疾病预测模型开发

文章目录 ML.NET库学习008&#xff1a;使用ML.NET进行心脏疾病预测模型开发1. 项目主要目的和原理2. 项目概述实现的主要功能&#xff1a;主要流程步骤&#xff1a;关键技术&#xff1a; 3. 主要功能和步骤数据加载与路径处理模型训练与评估模型保存与加载 4. 代码中的数据结构…...

了解rtc_time64_to_tm()和rtc_tm_to_time64()

rtc_time64_to_tm()和rtc_tm_to_time64()主要用于RTC的驱动程序&#xff0c;在Linux外部RTC驱动中较常见。 打开“drivers/rtc/lib.c” /* * rtc_time64_to_tm - Converts time64_t to rtc_time. * Convert seconds since 01-01-1970 00:00:00 to Gregorian date. */ //将ti…...

verilog程序设计及SystemVerilog验证

1.Verilog测试程序设计基础 1.1Testbench及其结构 在仿真的时候Testbench用来产生测试激励给待验证设计( Design Under Verification, DUV)&#xff0c;或者称为待测设计(Design UnderTest, DUT) 。 测试程序的一般结构&#xff1a; Testbench是一个测试平台&#xff0c;信号…...

智能编程助手功能革新与价值重塑之:GitHub Copilot

引言&#xff1a; GitHub Copilot 的最新更新为开发者带来了显著变化&#xff0c;其中 Agent Mode 功能尤为引人注目。该模式能够自动识别并修复代码错误、自动生成终端命令&#xff0c;并具备多级任务推理能力&#xff0c;这使得开发者在开发复杂功能时&#xff0c;可大幅减少…...

物联网行业通识:从入门到深度解析

物联网行业通识&#xff1a;从入门到深度解析 &#xff08;图1&#xff1a;物联网生态示意图&#xff09; 一、引言&#xff1a;万物互联时代的到来 根据IDC最新预测&#xff0c;到2025年全球物联网设备连接数将突破410亿&#xff0c;市场规模达1.1万亿美元。物联网&#xff…...

ABP - 事件总线之分布式事件总线

ABP - 事件总线之分布式事件总线 1. 分布式事件总线的集成1.2 基于 RabbitMQ 的分布式事件总线 2. 分布式事件总线的使用2.1 发布2.2 订阅2.3 事务和异常处理 3. 自己扩展的分布式事件总线实现 事件总线可以实现代码逻辑的解耦&#xff0c;使代码模块之间功能职责更清晰。而分布…...

【硬件设计细节】缓冲驱动器使用注意事项

一、缓冲驱动器核心功能与选型原则 信号增强与隔离 驱动能力匹配&#xff1a;根据负载电流需求选择缓冲器&#xff0c;例如CMOS缓冲器驱动能力通常为4-8mA&#xff0c;需搭配大电流负载时选用图腾柱输出或专用驱动芯片&#xff08;如TI的SN74LVC系列&#xff09;。电压域转换&…...

驱动开发系列38 - Linux Graphics 3D 绘制流程(一)- 创建画布

一:概述 当应用程序创建 OpenGL 上下文时,它通常需要申请帧缓冲(Framebuffer,即画布)。在 X11 体系下,应用程序不会直接向内核的 DRM 模块请求创建帧缓冲,而是通过 X 服务器进行申请。 虽然从技术上讲,应用程序可以直接使用 DRM 接口创建帧缓冲对象(BO),但为了将其与…...

再谈SpringCloud Gateway源码

再谈SpringCloud Gateway源码 一、整体请求流程二、前置对象准备1、实例化HandlerMapping2、实例化Route3、实例化WebHandler 三、实践业务扩展点1、定义扩展Route对象2、Filter能做什么3、定义扩展Filter对象4、定义父类Filter简化请求参数处理 前言&#xff1a; 之前有阅读过…...

把 CSV 文件摄入到 Elasticsearch 中 - CSVES

在我们之前的很多文章里&#xff0c;我有讲到这个话题。在今天的文章中&#xff0c;我们就提重谈。我们使用一种新的方法来实现。这是一个基于 golang 的开源项目。项目的源码在 https://github.com/githubesson/csves/。由于这个原始的代码并不支持 basic security 及带有安全…...