0

0

构建支持任意深度递归解析的 n 叉表达式树

霞舞

霞舞

发布时间:2026-01-06 16:15:42

|

531人浏览过

|

来源于php中文网

原创

构建支持任意深度递归解析的 n 叉表达式树

本文介绍如何通过递归下降解析器将嵌套括号表达式(如 `"(", "a", "&", "b", ")", "|", "c"`)准确构建成多叉树结构,自动处理 5 层甚至更深的嵌套,并避免手动拼接 `.nodes[x].nodes[y]` 等易错路径。

在构建逻辑表达式树(如 & 表示 AND、| 表示 OR)时,核心挑战并非节点存储本身,而是如何让程序自动“感知”当前解析深度,并在正确层级上创建/插入子节点。手动用 a.nodes[0].nodes[1].add_node(...) 显式访问多级路径不仅脆弱(索引越界、结构变更即崩溃),更违背树的抽象本质——树的结构应由语义驱动,而非硬编码路径。

解决方案是采用递归下降解析(Recursive Descent Parsing):将每个括号对 (...) 视为一个独立子表达式单元,其内部结构由递归调用自身完成;运算符(& / |)则自然升格为当前子树的根节点,其操作数成为子节点。该方法天然支持任意嵌套深度,无需预判层级数。

以下是一个精简、健壮的实现:

class Node:
    def __init__(self, val, nodes=None):
        self.val = val
        self.nodes = nodes or []

    def __repr__(self):
        return f"Node({self.val}): {self.nodes}"

def expr_to_tree(expr):
    it = iter(expr)  # 使用迭代器避免索引管理

    def get_operand():
        token = next(it, None)
        if token in (")", "&", "|", None):
            raise ValueError(f"Expected operand, got {repr(token)}")
        return expr_to_tree(expr) if token == "(" else Node(token)

    def get_expr(terminal=None):
        # 解析首个操作数(可能是原子值或子表达式)
        left = get_operand()

        # 尝试读取后续运算符
        op = next(it, None)
        if op not in ("&", "|"):
            # 无运算符:单节点表达式,直接返回
            if op != terminal:
                raise ValueError(f"Unexpected token {repr(op)}, expected {repr(terminal)}")
            return left

        # 有运算符:构造以该运算符为根的子树
        root = Node(op, [left])
        # 持续追加右侧操作数,直到遇到终止符(如 ')' 或结束)
        while True:
            right = get_operand()
            root.nodes.append(right)
            next_token = next(it, None)
            if next_token == terminal or next_token is None:
                return root
            elif next_token in ("&", "|"):
                if next_token != op:
                    raise ValueError(f"Mixed operators '{op}' and '{next_token}' not allowed")
                # 继续同级追加
                continue
            else:
                raise ValueError(f"Unexpected token {repr(next_token)} after operand")

    return get_expr()
✅ 关键设计说明:get_operand() 处理两种情况:遇到 "(" 则递归调用 expr_to_tree() 构建子树;否则创建叶子节点。get_expr() 负责识别运算符并聚合同级操作数,自动形成 Node('&'): [Node(A), Node(B), ...] 结构。迭代器 it 全局共享,确保每个递归层级按序消费输入,无需传递索引或手动切片。

使用示例:

Lovart
Lovart

全球首个AI设计智能体

下载
complicated_expr = ["(", "MORE", "&", "(", "COMPLICATED", "|","(","EXPRESSION","&","PRESENTING",")","|", "MANY", ")", ")", "|", "(", "DEEPER", "&", "(", "LEVELS", "|", "FOR", "|", "TREE", ")",")"]

tree = expr_to_tree(complicated_expr)
print(tree)
# 输出结构清晰,自动嵌套,无需任何 .nodes[0].nodes[1]... 手动寻址

⚠️ 注意事项

  • 此解析器要求表达式语法合法(括号匹配、运算符不混用),生产环境建议增加语法校验或错误恢复逻辑;
  • 若需支持优先级(如 & 优先于 |),需扩展为多层 get_expr()(如 get_or_expr() → get_and_expr() → get_operand());
  • 树节点未提供查找/修改接口,如需动态增删,应在 Node 类中补充 find() 或 insert_at_path() 方法(路径可用元组 (0,1,2) 表示)。

总结:面对深度不确定的 n 叉树构建,放弃“路径拼接”思维,拥抱递归解析——它让代码与问题结构对齐,将复杂度从开发者转移给调用,是处理嵌套语法的通用范式。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1462

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

227

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

85

2025.10.17

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1005

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

56

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

334

2025.12.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

380

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

567

2023.08.10

C++ 高性能计算与并行编程
C++ 高性能计算与并行编程

本专题专注于 C++ 在高性能计算(HPC)与并行编程中的应用,涵盖多线程、并发数据处理、OpenMP、MPI、GPU加速等技术。通过实际案例,帮助开发者掌握 如何利用 C++ 进行大规模数据计算和并行处理,提高程序的执行效率,适应高性能计算与数据密集型应用场景。

5

2026.01.08

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
HTML5/CSS3/JavaScript/ES6入门课程
HTML5/CSS3/JavaScript/ES6入门课程

共102课时 | 6.6万人学习

前端基础到实战(HTML5+CSS3+ES6+NPM)
前端基础到实战(HTML5+CSS3+ES6+NPM)

共162课时 | 18.6万人学习

第二十二期_前端开发
第二十二期_前端开发

共119课时 | 12.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号