全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

C++如何实现一个简单的AST_使用C++构建抽象语法树并进行代码解析

答案:文章介绍了在C++中构建简单抽象语法树(AST)的过程,涵盖节点基类定义、具体节点类型实现、变量环境管理、词法分析器与递归下降解析器的设计,并通过示例展示表达式解析与求值流程。

在C++中实现一个简单的抽象语法树(AST)是理解编译器或解释器工作原理的关键一步。AST 是源代码结构的树形表示,它忽略掉源码中的语法细节(如括号、分号),专注于程序的逻辑结构。下面我们将一步步构建一个极简的 AST,并结合词法分析和语法分析来解析简单表达式。

1. 定义AST节点基类

所有AST节点都应继承自一个公共基类。我们使用多态来处理不同类型的节点,比如数字、变量、二元操作等。

#include 
#include 
#include 
#include 

// 所有AST节点的基类 struct ExprAST { virtual ~ExprAST() = default; virtual double evaluate() const = 0; // 解释执行接口 };

2. 实现具体节点类型

根据常见表达式元素,定义几种基本节点:

  • 数字节点:表示常量值
  • 二元操作节点:如加减乘除
  • 变量节点:表示标识符
  • 赋值节点:将值绑定到变量

// 数字常量节点
struct NumberExprAST : ExprAST {
    double val;
    explicit NumberExprAST(double v) : val(v) {}
    double evaluate() const override { return val; }
};

// 变量节点 struct VariableExprAST : ExprAST { std::string name; explicit VariableExprAST(const std::string &n) : name(n) {} double evaluate() const override; };

// 二元操作节点(+、-、*、/) struct BinaryExprAST : ExprAST { char op; std::unique_ptr lhs, rhs;

BinaryExprAST(char o, std::unique_ptrzuojiankuohaophpcnExprASTyoujiankuohaophpcn l, std::unique_ptrzuojiankuohaophpcnExprASTyoujiankuohaophpcn r)
    : op(o), lhs(std::move(l)), rhs(std::move(r)) {}

double evaluate() const override;

};

// 赋值节点 struct AssignmentExprAST : ExprAST { std::string varName; std::unique_ptr expr;

AssignmentExprAST(const std::string &name, std::unique_ptrzuojiankuohaophpcnExprASTyoujiankuohaophpcn e)
    : varName(name), expr(std::move(e)) {}

double evaluate() const override;

};

这些节点通过重写 evaluate 方法实现求值逻辑。变量和赋值需要访问一个全局变量环境。

3. 添加变量环境支持

我们需要一个地方存储变量值。使用一个简单的 map 即可。

static std::map variableValues;

double VariableExprAST::evaluate() const { auto it = variableValues.find(name); if (it != variableValues.end()) return it->second; return 0.0; // 未定义变量默认为0 }

double AssignmentExprAST::evaluate() const { double val = expr->evaluate(); variableValues[varName] = val; return val; }

4. 简单的词法分析器(Tokenizer)

将输入字符串拆分为 token 流。这里只处理数字、字母、操作符和空格。

class Lexer {
    std::string input;
    size_t pos = 0;

public: explicit Lexer(const std::string &src) : input(src) {}

int getNextToken() {
    while (pos zuojiankuohaophpcn input.size() && isspace(input[pos]))
        pos++;

    if (pos >= input.size()) return 0; // EOF

    if (isdigit(input[pos]) || input[pos] == '.') {
        std::string numStr;
        while (pos zuojiankuohaophpcn input.size() && (isdigit(input[pos]) || input[pos] == '.'))
            numStr += input[pos++];
        lastNum = stod(numStr);
        return 'n'; // number token
    }

    if (isalpha(input[pos])) {
        std::string name;
        while (pos zuojiankuohaophpcn input.size() && isalnum(input[pos]))
            name += input[pos++];
        lastIdentifier = name;
        return 'v'; // variable or identifier
    }

    if (input[pos] == '=') {
        pos++;
        return '='; // assignment
    }

    return input[pos++]; // operator or punctuation
}

double lastNum;
std::string lastIdentifier;

};

5. 递归下降解析器

实现一个简单的递归下降解析器来构建AST。我们支持如下文法:

  • 表达式 → 赋值表达式
  • 赋值表达式 → 标识符 '=' 表达式 | 加减表达式
  • 加减表达式 → 乘除表达式的序列(用 + 或 - 连接)
  • 乘除表达式 → 原子表达式的序列(用 * 或 / 连接)
  • 原子表达式 → 数字 | 变量 | (表达式)

class Parser {
    Lexer lexer;
    int currentToken;
void advance() { currentToken = lexer.getNextToken(); }

std::unique_ptrzuojiankuohaophpcnExprASTyoujiankuohaophpcn parsePrimary();
std::unique_ptrzuojiankuohaophpcnExprASTyoujiankuohaophpcn parseExpression();
std::unique_ptrzuojiankuohaophpcnExprASTyoujiankuohaophpcn parseBinOpRHS(int precedence, std::unique_ptrzuojiankuohaophpcnExprASTyoujiankuohaophpcn lhs);
std::unique_ptrzuojiankuohaophpcnExprASTyoujiankuohaophpcn parseAssignment();

public: explicit Parser(const std::string &src) : lexer(src) { advance(); // 初始化第一个token }

std::unique_ptrzuojiankuohaophpcnExprASTyoujiankuohaophpcn parse();

};

std::unique_ptr Parser::parse() { if (currentToken == 0) return nullptr; return parseAssignment(); }

std::unique_ptr Parser::parseAssignment() { if (currentToken == 'v') { std::string idName = lexer.lastIdentifier; advance(); if (currentToken == '=') { advance(); auto expr = parseExpression(); if (!expr) return nullptr; return std::make_unique(idName, std::move(expr)); } // 不是赋值,则回退为普通变量使用 return std::make_unique(idName); } return parseExpression(); }

// 支持优先级的二元表达式解析 std::unique_ptr Parser::parseExpression() { auto lhs = parsePrimary(); if (!lhs) return nullptr; return parseBinOpRHS(0, std::move(lhs)); }

int getOperatorPrecedence(char op) { switch (op) { case '+': case '-': return 1; case '*': case '/': return 2; default: return -1; } }

std::unique_ptr Parser::parseBinOpRHS(int precedence, std::unique_ptr lhs) { while (true) { int prec = getOperatorPrecedence(currentToken); if (prec

    char op = currentToken;
    advance();

    auto rhs = parsePrimary();
    if (!rhs) return nullptr;

    int nextPrec = getOperatorPrecedence(currentToken);
    if (nextPrec > prec) {
        rhs = parseBinOpRHS(prec + 1, std::move(rhs));
        if (!rhs) return nullptr;
    }

    lhs = std::make_uniquezuojiankuohaophpcnBinaryExprASTyoujiankuohaophpcn(op, std::move(lhs), std::move(rhs));
}

}

std::unique_ptr Parser::parsePrimary() { switch (currentToken) { case 'n': { auto result = std::make_unique(lexer.lastNum); advance(); return result; } case 'v': { auto result = std::make_unique(lexer.lastIdentifier); advance(); return result; } case '(': { advance(); // consume '(' auto expr = parseExpression(); if (currentToken != ')') { std::cerr

// 二元操作求值实现 double BinaryExprAST::evaluate() const { double L = lhs->evaluate(); double R = rhs->evaluate(); switch (op) { case '+': return L + R; case '-': return L - R; case '': return L R; case '/': return L / R; default: return 0.0; } }

6. 使用示例

现在我们可以测试整个流程:

int main() {
    std::string input;
    std::cout << "Enter expression (e.g., a=3+4*2): ";
    std::getline(std::cin, input);
Parser parser(input);
auto ast = parser.parse();

if (ast) {
    double result = ast-youjiankuohaophpcnevaluate();
    std::cout zuojiankuohaophpcnzuojiankuohaophpcn "Result: " zuojiankuohaophpcnzuojiankuohaophpcn result zuojiankuohaophpcnzuojiankuohaophpcn "\n";
    std::cout zuojiankuohaophpcnzuojiankuohaophpcn "Variables:\n";
    for (const auto& [name, val] : variableValues) {
        std::cout zuojiankuohaophpcnzuojiankuohaophpcn "  " zuojiankuohaophpcnzuojiankuohaophpcn name zuojiankuohaophpcnzuojiankuohaophpcn " = " zuojiankuohaophpcnzuojiankuohaophpcn val zuojiankuohaophpcnzuojiankuohaophpcn "\n";
    }
} else {
    std::cerr zuojiankuohaophpcnzuojiankuohaophpcn "Parse error.\n";
}

return 0;

}

运行示例:

输入:a=3+4*2
输出:Result: 11
      Variables: a = 11

这个例子展示了如何从零开始构建一个可运行的 AST 系统。虽然功能简单,但它具备了真实编译器前端的核心组件:词法分析、语法分析、AST 构建与解释执行。

基本上就这些。你可以在此基础上扩展函数定义、控制流语句(if、while)、作用域管理等功能,逐步演化成一个完整的解释型语言前端。


# c++  # 前端  # git  # 字节  # ai  # ios  # switch  # stream  # 作用域  # String  # 常量  # if  # while  # 多态  # Token  # 标识符  # const  # auto  # 全局变量  # 字符串  # 递归  # char  # int  # double  # cerr  # 继承  # public  # Struct  # map  # default  # 求值  # 加减  # 构建一个  # 加减乘除  # 你可以  # 第一个  # 我们可以  # 几种  # 环境管理 


相关文章: 简单实现Android文件上传  php8.4新语法match怎么用_php8.4match表达式替代switch【方法】  如何在建站主机中优化服务器配置?  如何用5美元大硬盘VPS安全高效搭建个人网站?  建站之星如何防范黑客攻击与数据泄露?  Android自定义listview布局实现上拉加载下拉刷新功能  c# F# 的 MailboxProcessor 和 C# 的 Actor 模型  如何通过老薛主机一键快速建站?  青岛网站设计制作公司,查询青岛招聘信息的网站有哪些?  湖北网站制作公司有哪些,湖北清能集团官网?  如何通过主机屋免费建站教程十分钟搭建网站?  c# 在ASP.NET Core中管理和取消后台任务  制作网站公司那家好,网络公司是做什么的?  交易网站制作流程,我想开通一个网站,注册一个交易网址,需要那些手续?  网站制作公司,橙子建站是合法的吗?  官网建站费用明细查询_企业建站套餐价格及收费标准指南  微信h5制作网站有哪些,免费微信H5页面制作工具?  标准网站视频模板制作软件,现在有哪个网站的视频编辑素材最齐全的,背景音乐、音效等?  c# 在高并发场景下,委托和接口调用的性能对比  如何在万网开始建站?分步指南解析  百度网页制作网站有哪些,谁能告诉我百度网站是怎么联系?  常州自助建站:操作简便模板丰富,企业个人快速搭建网站  如何配置支付宝与微信支付功能?  建站之星后台管理:高效配置与模板优化提升用户体验  深圳网站制作培训,深圳哪些招聘网站比较好?  如何用PHP快速搭建CMS系统?  宝塔新建站点报错如何解决?  安云自助建站系统如何快速提升SEO排名?  制作网站的模板软件,网站怎么建设?  如何通过建站之星自助学习解决操作问题?  logo在线制作免费网站在线制作好吗,DW网页制作时,如何在网页标题前加上logo?  如何通过山东自助建站平台快速注册域名?  猪八戒网站制作视频,开发一个猪八戒网站,大约需要多少?或者自己请程序员,需要什么程序员,多少程序员能完成?  无锡制作网站公司有哪些,无锡优八网络科技有限公司介绍?  专业公司网站制作公司,用什么语言做企业网站比较好?  广州建站公司哪家好?十大优质服务商推荐  ,石家庄四十八中学官网?  建站之星导航如何优化提升用户体验?  建站OpenVZ教程与优化策略:配置指南与性能提升  怎么制作一个起泡网,水泡粪全漏粪育肥舍冬季氨气超过25ppm,可以有哪些措施降低舍内氨气水平?  建站主机如何安装配置?新手必看操作指南  定制建站流程解析:需求评估与SEO优化功能开发指南  深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?  如何挑选高效建站主机与优质域名?  微网站制作教程,不会写代码,不会编程,怎么样建自己的网站?  5种Android数据存储方式汇总  网站制作大概要多少钱一个,做一个平台网站大概多少钱?  利用JavaScript实现拖拽改变元素大小  音响网站制作视频教程,隆霸音响官方网站?  c++怎么编写动态链接库dll_c++ __declspec(dllexport)导出与调用【方法】 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。