Java中基于自定义链表栈的括号平衡性检查指南

Java中基于自定义链表栈的括号平衡性检查指南

本教程深入探讨了如何利用自定义实现的链表来高效、准确地判断括号表达式的平衡性。文章首先剖析了传统两栈方法的不足,随后详细阐述了业界普遍采用的单栈算法原理,并提供了完整的java代码实现及使用示例。通过本指南,读者将掌握栈在解决结构匹配问题中的核心应用,并能构建健壮的括号平衡性检查逻辑。

引言:括号平衡性与栈的重要性

计算机科学中,括号平衡性是一个基础而重要的问题,常见于编译器、解释器和代码分析工具中。一个括号表达式被认为是平衡的,当且仅当它满足以下条件:

表达式中每个开括号(如 ()都有一个对应的闭括号(如 ))。括号的嵌套顺序是正确的,即任何一个闭括号都必须与其最近的未闭合的开括号相匹配。

例如,((())) 和 () 是平衡的,而 (()、)( 和 ()) 则是不平衡的。栈(Stack)作为一种“后进先出”(LIFO)的数据结构,天然适用于解决这类需要匹配和回溯的问题。

问题剖析:两栈法的局限性

在最初的尝试中,可能有人会想到使用两个栈来解决括号平衡问题:一个栈用于存储开括号,另一个栈用于存储闭括号。然后,通过比较两个栈中元素的数量来判断是否平衡。然而,这种方法存在根本性的缺陷。

考虑以下原始代码片段中的 isBalanced 方法:

立即学习“Java免费学习笔记(深入)”;

static boolean isBalanced(String expr){    if (expr == null || expr.length() % 2 == 1) {        return false;    }    Stack stack1 = new Stack(); // 存储开括号    Stack stack2 = new Stack(); // 存储闭括号    // 遍历表达式,将开括号和闭括号分别压入不同的栈    for (int i = 0; i< expr.length(); i++){        if (expr.charAt(i) == '(') {            stack1.push(expr.charAt(i));        }        if (expr.charAt(i) == ')') {           stack2.push(expr.charAt(i));        }    }    // 尝试弹出所有元素    for(int i = 0; i< expr.length(); i++) {      stack1.pop();      stack2.pop();    }    return (stack1.isEmpty() && stack2.isEmpty()) ;}

这种方法的缺点主要体现在:

无法检查嵌套顺序:它只能保证开括号和闭括号的数量相等,但无法判断它们的出现顺序是否正确。例如,对于表达式 )(,stack1 会包含 (,stack2 会包含 )。最终两者都为空,但 )( 显然是不平衡的。不必要的弹出操作:第二个 for 循环尝试无差别地弹出 expr.length() 次元素。这可能导致在栈中元素不足时,反复调用 pop() 方法,从而触发“Trying to pop when stack is empty”的错误提示,即使最终结果可能碰巧是 true 或 false,也掩盖了潜在的逻辑错误。正确的做法应该是在弹出前检查栈是否为空,并且弹出的次数应该与栈中实际的元素数量相关,而非表达式的总长度。

因此,这种两栈分离、独立计数的策略并不能有效地解决括号平衡性问题。我们需要一种能够实时匹配括号的机制。

核心算法:单栈法实现括号平衡性检查

解决括号平衡问题的标准方法是使用一个单一的栈。其核心思想是:当遇到开括号时,将其压入栈中;当遇到闭括号时,检查栈顶元素是否为对应的开括号。

以下是单栈算法的详细步骤:

预处理与边界条件检查

如果表达式为 null 或其长度为奇数,则直接判断为不平衡(因为平衡的括号表达式长度必须为偶数)。如果表达式为空字符串 “”,通常认为它是平衡的。

遍历表达式:从左到右逐个字符地扫描输入表达式。

处理开括号:如果当前字符是一个开括号(如 (),则将其压入栈中。

处理闭括号:如果当前字符是一个闭括号(如 )):

检查栈是否为空:如果此时栈为空,说明没有对应的开括号可供匹配,因此表达式不平衡,立即返回 false。弹出并匹配:如果栈不为空,则从栈中弹出一个元素。这个弹出的元素应该是一个开括号,并且必须与当前遇到的闭括号类型相匹配。对于本例中只有 () 一种括号的情况,弹出的必须是 (。如果弹出的不是 (,则表示括号不匹配,表达式不平衡,立即返回 false。

最终检查:在遍历完整个表达式后:

如果栈为空,表示所有开括号都找到了对应的闭括号并成功匹配,表达式是平衡的。如果栈不为空,表示仍有未匹配的开括号(即多余的开括号),表达式不平衡。

根据上述算法,我们可以重构 isBalanced 方法,并结合自定义的 Stack 类实现:

public class BalanceChecker {    // 假设 Stack 和 Node 类已在同一包中定义,且不允许导入其他库    /**     * 检查给定字符串表达式中的括号是否平衡。     * 仅支持圆括号 ()。     *     * @param expr 待检查的字符串表达式。     * @return 如果括号平衡则返回 true,否则返回 false。     */    static boolean isBalanced(String expr) {        // 初始检查:null或奇数长度的表达式必然不平衡        if (expr == null || expr.length() % 2 != 0) {            return false;        }        // 对于空字符串,通常认为是平衡的        if (expr.isEmpty()) {            return true;        }        Stack stack = new Stack(); // 使用单个栈        // 遍历输入表达式        for (int i = 0; i < expr.length(); i++) {            char current = expr.charAt(i);            // 如果是开括号,则压入栈中            if (current == '(') {                stack.push(current);            }            // 如果是闭括号            else if (current == ')') {                // 如果栈为空,说明没有匹配的开括号,不平衡                if (stack.isEmpty()) {                    return false;                }                // 弹出栈顶元素,检查是否匹配                // pop() 方法返回 Object,需要强制转换为 char                char topChar = (char) stack.pop();                // 对于只有一种括号的情况,这一步的匹配检查是隐式的                // 因为我们只压入 '(',所以如果弹出不是 '(',说明逻辑有问题                // 但为了通用性,保留此检查                if (topChar != '(') {                    return false; // 弹出的不是对应的开括号,则不平衡                }            }            // 假设表达式只包含括号,如果遇到其他字符,可以根据需求处理            // 例如,可以忽略,或者直接返回 false (视为无效字符)        }        // 遍历结束后,如果栈为空,则所有开括号都有匹配的闭括号,表达式平衡        return stack.isEmpty();    }    public static void main(String[] args) {        System.out.println("测试用例:");        System.out.println("((())) is balanced: " + isBalanced("((()))")); // true        System.out.println("() is balanced: " + isBalanced("()"));         // true        System.out.println("(() is balanced: " + isBalanced("(()"));       // false (多余的开括号)        System.out.println(")() is balanced: " + isBalanced(")()"));       // false (开局闭括号)        System.out.println(")( is balanced: " + isBalanced(")("));         // false (顺序错误)        System.out.println("'' is balanced: " + isBalanced(""));           // true (空字符串)        System.out.println("null is balanced: " + isBalanced(null));       // false (null字符串)        System.out.println("(( is balanced: " + isBalanced("(("));         // false (多余的开括号)        System.out.println(")) is balanced: " + isBalanced("))"));         // false (多余的闭括号)        System.out.println("())(() is balanced: " + isBalanced("())(()")); // false (复杂不平衡)    }}

自定义栈与节点类的实现

由于题目要求不允许导入任何库,我们需要使用自定义的 Stack 和 Node 类。这些类通常通过链表结构实现,以提供动态大小的栈。

Node 类

Node 类是链表的基本组成单元,每个节点包含数据 (info) 和指向下一个节点的引用 (next)。

public class Node {    Object info; // 存储节点信息    Node next;   // 指向链表中的下一个节点    /**     * 构造函数,创建一个新的节点。     * @param info 节点中存储的数据。     * @param next 指向下一个节点的引用。     */    Node(Object info, Node next){        this.info = info;        this.next = next;    }}

Stack 类

Stack 类基于 Node 类实现了一个链表结构的栈,遵循 LIFO(Last-In, First-Out)原则。栈的顶部由 top 引用指示。

public class Stack {    private Node top; // 栈顶元素的引用    /**     * 构造函数,创建一个空栈。     */    public Stack() {        top = null;    }    /**     * 检查栈是否为空。     * @return 如果栈为空则返回 true,否则返回 false。     */    public boolean isEmpty(){        return (top == null);    }    /**     * 将一个新元素压入栈顶。     * @param newItem 要压入栈的元素。     */    public void push(Object newItem){        top = new Node(newItem, top); // 新元素成为新的栈顶,并指向原来的栈顶    }    /**     * 从栈顶弹出一个元素。     * @return 栈顶元素。如果栈为空,则打印错误信息并返回 null。     */    public Object pop(){        if (isEmpty()){            System.out.println("Trying to pop when stack is empty");            return null;        } else {            Node temp = top; // 保存当前栈顶节点            top = top.next;  // 栈顶下移            return temp.info; // 返回原栈顶节点的数据        }    }    /**     * 清空栈中的所有元素。     */    void popAll(){        top = null;    }    /**     * 查看栈顶元素但不将其移除。     * @return 栈顶元素。如果栈为空,则打印错误信息并返回 null。     */    public Object peek(){        if (isEmpty()){           System.out.println("Trying to peek when stack is empty");           return null;        } else {           return top.info; // 返回栈顶元素的数据        }    }} // End of Stack using a linked list

示例与注意事项

示例测试

为了验证 isBalanced 方法的正确性,我们可以使用 main 方法进行测试:

// 包含在 BalanceChecker 类中的 main 方法public static void main(String[] args) {    System.out.println("测试用例:");    System.out.println("((())) is balanced: " + isBalanced("((()))")); // true    System.out.println("() is balanced: " + isBalanced("()"));         // true    System.out.println("(() is balanced: " + isBalanced("(()"));       // false (多余的开括号)    System.out.println(")() is balanced: " + isBalanced(")()"));       // false (开局闭括号)    System.out.println(")( is balanced: " + isBalanced(")("));         // false (顺序错误)    System.out.println("'' is balanced: " + isBalanced(""));           // true (空字符串)    System.out.println("null is balanced: " + isBalanced(null));       // false (null字符串)    System.out.println("(( is balanced: " + isBalanced("(("));         // false (多余的开括号)    System.out.println(")) is balanced: " + isBalanced("))"));         // false (多余的闭括号)    System.out.println("())(() is balanced: " + isBalanced("())(()")); // false (复杂不平衡)}

注意事项

无导入限制:严格遵守不允许导入任何Java标准库的限制,所有数据结构(如栈)都必须是自定义实现。类型转换:Stack 类的 push 和 pop 方法处理的是 Object 类型。在 isBalanced 方法中,当从栈中弹出字符时,需要进行显式的 (char) stack.pop() 类型转换。扩展性:当前 isBalanced 方法仅支持圆括号 ()。如果需要支持多种括号类型(如 {}, []),则需要在 if-else if 结构中增加对这些括号的判断,并在弹出时确保匹配的是正确的开括号。例如,遇到 ] 时,栈顶必须是 [。错误处理:自定义 Stack 的 pop() 和 peek() 方法在栈为空时会打印错误信息并返回 null。在 isBalanced 方法中,我们通过 stack.isEmpty() 提前检查来避免这种情况,这是更健壮的做法。效率:单栈算法的时间复杂度为 O(n),其中 n 是表达式的长度,因为它只需要对表达式进行一次遍历。空间复杂度为 O(n),最坏情况下(所有字符都是开括号)栈会存储所有开括号。

总结

本教程详细阐述了使用自定义链表栈来检查括号表达式平衡性的正确方法。通过对比分析两栈法的局限性,我们强调了单栈算法在处理括号嵌套和匹配方面的优越性。掌握这种利用栈解决结构匹配问题的能力,对于理解和编写更复杂的解析器和验证逻辑至关重要。遵循本指南提供的代码示例和注意事项,开发者可以构建出高效、健壮的括号平衡性检查机制。

以上就是Java中基于自定义链表栈的括号平衡性检查指南的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/21914.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月1日 15:54:09
下一篇 2025年11月1日 16:09:28

相关推荐

  • soul怎么发长视频瞬间_Soul长视频瞬间发布方法

    可通过分段发布、格式转换或剪辑压缩三种方法在Soul上传长视频。一、将长视频用相册编辑功能拆分为多个30秒内片段,依次发布并标注“Part 1”“Part 2”保持连贯;二、使用“格式工厂”等工具将视频转为MP4(H.264)、分辨率≤1080p、帧率≤30fps、大小≤50MB,适配平台要求;三、…

    2025年12月6日 软件教程
    500
  • 天猫app淘金币抵扣怎么使用

    在天猫app购物时,淘金币是一项能够帮助你节省开支的实用功能。掌握淘金币的抵扣使用方法,能让你以更实惠的价格买到心仪商品。 当你选好商品并准备下单时,记得查看商品页面是否支持淘金币抵扣。如果该商品支持此项功能,在提交订单的页面会明确显示相关提示。你会看到淘金币的具体抵扣比例——通常情况下,淘金币可按…

    2025年12月6日 软件教程
    500
  • Pboot插件缓存机制的详细解析_Pboot插件缓存清理的命令操作

    插件功能异常或页面显示陈旧内容可能是缓存未更新所致。PbootCMS通过/runtime/cache/与/runtime/temp/目录缓存插件配置、模板解析结果和数据库查询数据,提升性能但影响调试。解决方法包括:1. 手动删除上述目录下所有文件;2. 后台进入“系统工具”-“缓存管理”,勾选插件、…

    2025年12月6日 软件教程
    100
  • Word2013如何插入SmartArt图形_Word2013SmartArt插入的视觉表达

    答案:可通过四种方法在Word 2013中插入SmartArt图形。一、使用“插入”选项卡中的“SmartArt”按钮,选择所需类型并插入;二、从快速样式库中选择常用模板如组织结构图直接应用;三、复制已有SmartArt图形到目标文档后调整内容与格式;四、将带项目符号的文本选中后右键转换为Smart…

    2025年12月6日 软件教程
    000
  • 《kk键盘》一键发图开启方法

    如何在kk键盘中开启一键发图功能? 1、打开手机键盘,找到并点击“kk”图标。 2、进入工具菜单后,选择“一键发图”功能入口。 3、点击“去开启”按钮,跳转至无障碍服务设置页面。 4、在系统通用设置中,进入“已下载的应用”列表。 j2me3D游戏开发简单教程 中文WORD版 本文档主要讲述的是j2m…

    2025年12月6日 软件教程
    100
  • 怎样用免费工具美化PPT_免费美化PPT的实用方法分享

    利用KIMI智能助手可免费将PPT美化为科技感风格,但需核对文字准确性;2. 天工AI擅长优化内容结构,提升逻辑性,适合高质量内容需求;3. SlidesAI支持语音输入与自动排版,操作便捷,利于紧急场景;4. Prezo提供多种模板,自动生成图文并茂幻灯片,适合学生与初创团队。 如果您有一份内容完…

    2025年12月6日 软件教程
    000
  • Pages怎么协作编辑同一文档 Pages多人实时协作的流程

    首先启用Pages共享功能,点击右上角共享按钮并选择“添加协作者”,设置为可编辑并生成链接;接着复制链接通过邮件或社交软件发送给成员,确保其使用Apple ID登录iCloud后即可加入编辑;也可直接在共享菜单中输入邮箱地址定向邀请,设定编辑权限后发送;最后在共享面板中管理协作者权限,查看实时在线状…

    2025年12月6日 软件教程
    100
  • 哔哩哔哩的视频卡在加载中怎么办_哔哩哔哩视频加载卡顿解决方法

    视频加载停滞可先切换网络或重启路由器,再清除B站缓存并重装应用,接着调低播放清晰度并关闭自动选分辨率,随后更改播放策略为AVC编码,最后关闭硬件加速功能以恢复播放。 如果您尝试播放哔哩哔哩的视频,但进度条停滞在加载状态,无法继续播放,这通常是由于网络、应用缓存或播放设置等因素导致。以下是解决此问题的…

    2025年12月6日 软件教程
    000
  • REDMI K90系列正式发布,售价2599元起!

    10月23日,redmi k90系列正式亮相,推出redmi k90与redmi k90 pro max两款新机。其中,redmi k90搭载骁龙8至尊版处理器、7100mah大电池及100w有线快充等多项旗舰配置,起售价为2599元,官方称其为k系列迄今为止最完整的标准版本。 图源:REDMI红米…

    2025年12月6日 行业动态
    200
  • 买家网购苹果手机仅退款不退货遭商家维权,法官调解后支付货款

    10 月 24 日消息,据央视网报道,近年来,“仅退款”服务逐渐成为众多网购平台的常规配置,但部分消费者却将其当作“免费试用”的手段,滥用规则谋取私利。 江苏扬州市民李某在某电商平台购买了一部苹果手机,第二天便以“不想要”为由在线申请“仅退款”,当时手机尚在物流运输途中。第三天货物送达后,李某签收了…

    2025年12月6日 行业动态
    000
  • Linux中如何安装Nginx服务_Linux安装Nginx服务的完整指南

    首先更新系统软件包,然后通过对应包管理器安装Nginx,启动并启用服务,开放防火墙端口,最后验证欢迎页显示以确认安装成功。 在Linux系统中安装Nginx服务是搭建Web服务器的第一步。Nginx以高性能、低资源消耗和良好的并发处理能力著称,广泛用于静态内容服务、反向代理和负载均衡。以下是在主流L…

    2025年12月6日 运维
    000
  • 当贝X5S怎样看3D

    当贝X5S观看3D影片无立体效果时,需开启3D模式并匹配格式:1. 播放3D影片时按遥控器侧边键,进入快捷设置选择3D模式;2. 根据片源类型选左右或上下3D格式;3. 可通过首页下拉进入电影专区选择3D内容播放;4. 确认片源为Side by Side或Top and Bottom格式,并使用兼容…

    2025年12月6日 软件教程
    100
  • Linux journalctl与systemctl status结合分析

    先看 systemctl status 确认服务状态,再用 journalctl 查看详细日志。例如 nginx 启动失败时,systemctl status 显示 Active: failed,journalctl -u nginx 发现端口 80 被占用,结合两者可快速定位问题根源。 在 Lin…

    2025年12月6日 运维
    100
  • 华为新机发布计划曝光:Pura 90系列或明年4月登场

    近日,有数码博主透露了华为2025年至2026年的新品规划,其中pura 90系列预计在2026年4月发布,有望成为华为新一代影像旗舰。根据路线图,华为将在2025年底至2026年陆续推出mate 80系列、折叠屏新机mate x7系列以及nova 15系列,而pura 90系列则将成为2026年上…

    2025年12月6日 行业动态
    100
  • TikTok视频无法下载怎么办 TikTok视频下载异常修复方法

    先检查链接格式、网络设置及工具版本。复制以https://www.tiktok.com/@或vm.tiktok.com开头的链接,删除?后参数,尝试短链接;确保网络畅通,可切换地区节点或关闭防火墙;更新工具至最新版,优先选用yt-dlp等持续维护的工具。 遇到TikTok视频下载不了的情况,别急着换…

    2025年12月6日 软件教程
    100
  • Linux如何防止缓冲区溢出_Linux防止缓冲区溢出的安全措施

    缓冲区溢出可通过栈保护、ASLR、NX bit、安全编译选项和良好编码实践来防范。1. 使用-fstack-protector-strong插入canary检测栈破坏;2. 启用ASLR(kernel.randomize_va_space=2)随机化内存布局;3. 利用NX bit标记不可执行内存页…

    2025年12月6日 运维
    000
  • 2025年双十一买手机选直板机还是选折叠屏?建议看完这篇再做决定

    随着2025年双十一购物节的临近,许多消费者在选购智能手机时都会面临一个共同的问题:是选择传统的直板手机,还是尝试更具科技感的折叠屏设备?其实,这个问题的答案早已在智能手机行业的演进中悄然浮现——如今的手机市场已不再局限于“拼参数、堆配置”的初级竞争,而是迈入了以形态革新驱动用户体验升级的新时代。而…

    2025年12月6日 行业动态
    000
  • Linux如何优化系统性能_Linux系统性能优化的实用方法

    优化Linux性能需先监控资源使用,通过top、vmstat等命令分析负载,再调整内核参数如TCP优化与内存交换,结合关闭无用服务、选用合适文件系统与I/O调度器,持续按需调优以提升系统效率。 Linux系统性能优化的核心在于合理配置资源、监控系统状态并及时调整瓶颈环节。通过一系列实用手段,可以显著…

    2025年12月6日 运维
    000
  • Pboot插件数据库连接的配置教程_Pboot插件数据库备份的自动化脚本

    首先配置PbootCMS数据库连接参数,确保插件正常访问;接着创建auto_backup.php脚本实现备份功能;然后通过Windows任务计划程序或Linux Cron定时执行该脚本,完成自动化备份流程。 如果您正在开发或维护一个基于PbootCMS的网站,并希望实现插件对数据库的连接配置以及自动…

    2025年12月6日 软件教程
    000
  • 今日头条官方主页入口 今日头条平台直达网址官方链接

    今日头条官方主页入口是www.toutiao.com,该平台通过个性化信息流推送图文、短视频等内容,具备分类导航、便捷搜索及跨设备同步功能。 今日头条官方主页入口在哪里?这是不少网友都关注的,接下来由PHP小编为大家带来今日头条平台直达网址官方链接,感兴趣的网友一起随小编来瞧瞧吧! www.tout…

    2025年12月6日 软件教程
    000

发表回复

登录后才能评论
关注微信