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

在C++中实现一个简单的抽象语法树(AST)是理解编译器或解释器工作原理的关键一步。AST 是源代码结构的树形表示,它忽略掉源码中的语法细节(如括号、分号),专注于程序的逻辑结构。下面我们将一步步构建一个极简的 AST,并结合词法分析和语法分析来解析简单表达式。
1. 定义AST节点基类
所有AST节点都应继承自一个公共基类。我们使用多态来处理不同类型的节点,比如数字、变量、二元操作等。
#include <memory>
#include <string>
#include <map>
#include <iostream>
<p>// 所有AST节点的基类
struct ExprAST {
virtual ~ExprAST() = default;
virtual double evaluate() const = 0; // 解释执行接口
};</p>2. 实现具体节点类型
根据常见表达式元素,定义几种基本节点:
- 数字节点:表示常量值
- 二元操作节点:如加减乘除
- 变量节点:表示标识符
- 赋值节点:将值绑定到变量
// 数字常量节点
struct NumberExprAST : ExprAST {
double val;
explicit NumberExprAST(double v) : val(v) {}
double evaluate() const override { return val; }
};
<p>// 变量节点
struct VariableExprAST : ExprAST {
std::string name;
explicit VariableExprAST(const std::string &n) : name(n) {}
double evaluate() const override;
};</p><p>// 二元操作节点(+、-、*、/)
struct BinaryExprAST : ExprAST {
char op;
std::unique_ptr<ExprAST> lhs, rhs;</p><pre class='brush:php;toolbar:false;'>BinaryExprAST(char o, std::unique_ptr<ExprAST> l, std::unique_ptr<ExprAST> r)
: op(o), lhs(std::move(l)), rhs(std::move(r)) {}
double evaluate() const override;};
// 赋值节点
struct AssignmentExprAST : ExprAST {
std::string varName;
std::unique_ptr
AssignmentExprAST(const std::string &name, std::unique_ptr<ExprAST> e)
: varName(name), expr(std::move(e)) {}
double evaluate() const override;};
这些节点通过重写 evaluate 方法实现求值逻辑。变量和赋值需要访问一个全局变量环境。
3. 添加变量环境支持
我们需要一个地方存储变量值。使用一个简单的 map 即可。
static std::map<std::string, double> variableValues;
<p>double VariableExprAST::evaluate() const {
auto it = variableValues.find(name);
if (it != variableValues.end()) return it->second;
return 0.0; // 未定义变量默认为0
}</p><p>double AssignmentExprAST::evaluate() const {
double val = expr->evaluate();
variableValues[varName] = val;
return val;
}</p>4. 简单的词法分析器(Tokenizer)
将输入字符串拆分为 token 流。这里只处理数字、字母、操作符和空格。
class Lexer {
std::string input;
size_t pos = 0;
<p>public:
explicit Lexer(const std::string &src) : input(src) {}</p><pre class='brush:php;toolbar:false;'>int getNextToken() {
while (pos < input.size() && isspace(input[pos]))
pos++;
if (pos >= input.size()) return 0; // EOF
if (isdigit(input[pos]) || input[pos] == '.') {
std::string numStr;
while (pos < 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 < 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。我们支持如下文法:
GoEnhance
全能AI视频制作平台:通过GoEnhance AI让视频创作变得比以往任何时候都更简单。
347
查看详情
- 表达式 → 赋值表达式
- 赋值表达式 → 标识符 '=' 表达式 | 加减表达式
- 加减表达式 → 乘除表达式的序列(用 + 或 - 连接)
- 乘除表达式 → 原子表达式的序列(用 * 或 / 连接)
- 原子表达式 → 数字 | 变量 | (表达式)
class Parser {
Lexer lexer;
int currentToken;
<pre class='brush:php;toolbar:false;'>void advance() { currentToken = lexer.getNextToken(); }
std::unique_ptr<ExprAST> parsePrimary();
std::unique_ptr<ExprAST> parseExpression();
std::unique_ptr<ExprAST> parseBinOpRHS(int precedence, std::unique_ptr<ExprAST> lhs);
std::unique_ptr<ExprAST> parseAssignment();public: explicit Parser(const std::string &src) : lexer(src) { advance(); // 初始化第一个token }
std::unique_ptr<ExprAST> parse();
};
std::unique_ptr
std::unique_ptr
// 支持优先级的二元表达式解析
std::unique_ptr
int getOperatorPrecedence(char op) { switch (op) { case '+': case '-': return 1; case '*': case '/': return 2; default: return -1; } }
std::unique_ptr } std::unique_ptr // 二元操作求值实现
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;
}
} 现在我们可以测试整个流程: } 运行示例: 这个例子展示了如何从零开始构建一个可运行的 AST 系统。虽然功能简单,但它具备了真实编译器前端的核心组件:词法分析、语法分析、AST 构建与解释执行。 基本上就这些。你可以在此基础上扩展函数定义、控制流语句(if、while)、作用域管理等功能,逐步演化成一个完整的解释型语言前端。 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_unique<BinaryExprAST>(op, std::move(lhs), std::move(rhs));
}6. 使用示例
int main() {
std::string input;
std::cout << "Enter expression (e.g., a=3+4*2): ";
std::getline(std::cin, input);
<pre class='brush:php;toolbar:false;'>Parser parser(input);
auto ast = parser.parse();
if (ast) {
double result = ast->evaluate();
std::cout << "Result: " << result << "\n";
std::cout << "Variables:\n";
for (const auto& [name, val] : variableValues) {
std::cout << " " << name << " = " << val << "\n";
}
} else {
std::cerr << "Parse error.\n";
}
return 0;
输入:a=3+4*2
输出:Result: 11
Variables: a = 11
以上就是C++如何实现一个简单的AST_使用C++构建抽象语法树并进行代码解析的详细内容,更多请关注其它相关文章!
# 多态
# 莆田鞋营销推广话术模板
# 百度关键词排名找谁查看
# 商业网站建设pdf
# 黔东模板网站优化平台
# 南坪正规seo公司
# 长垣网站seo优化推广
# SEO首页是啥意思
# 焦作搜索关键词排名精准
# 增城企业网站seo
# 会展营销推广软文
# 构建一个
# 加减
# 求值
# 解决方法
# c++
# 全局变量
# 如何实现
# 有什么
# 重写
# 递归
# 作用域
# stream
# switch
# ios
# ai
# 字节
# git
# 前端
# 抽象语法树
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
Golang如何使用new_Go new分配内存机制讲解
poki网页游戏推荐_poki免费游戏平台入口
12306怎么选座位选到安静区_12306选座安静区域选择策略
小猿搜题在线学习页面在哪_小猿搜题在线学习中心入口
谷歌邮箱网页版官方页面入口 谷歌邮箱网页端快速访问
Surface怎么安装系统 微软Surface Pro U盘重装win11教程
自定义Bag-of-Words实现:处理带负号的词汇权重
如何在复杂的电商平台中优雅地管理共享资源并确保正确重定向,使用spryker-shop/resource-share-page模块助你一臂之力
怎么在浏览器上运行HTML文件_浏览器运行HTML文件技巧【技巧】
百度网盘网页版入口 百度网盘网页版官方登录网址
PyTorch模型训练效果不佳?深入剖析常见错误与调试技巧
高德地图怎么看全景照片_高德地图全景照片浏览教程
Golang如何优雅处理error_Golang error处理最佳实践总结
快手官方唯一登录入口 谨防山寨钓鱼网站
win11开机启动修复循环怎么办 Win11无法进入系统高级启动解决方法【修复】
QQ邮箱官方登录入口_QQ邮箱网页版快捷使用平台
Centos/Linux 系统下安装 composer 的完整步骤
c++中的std::forward_list和std::list有什么不同_c++ forward_list与list区别分析
微信群消息显示延迟如何解决 微信群消息刷新优化方法
css子元素高度不一致导致布局错位怎么办_使用align-items:stretch解决高度差异
Lar*el的路由模型绑定怎么用_Lar*el Route Model Binding简化控制器逻辑
深入理解J*a合成构造器:何时以及为何阻止其生成
html网页设计源代码怎么运行_运行html网页设计源代码步骤【指南】
163邮箱登录密码 163邮箱忘记密码找回
CSS布局:解决全屏元素100%尺寸与外边距导致的页面溢出问题
美团外卖商家服务中心入口 美团商家版官网入口
MAC怎么安装Homebrew包管理器_MAC为开发者和高级用户安装命令行工具
Sublime Text怎么显示空格和制表符_Sublime显示不可见字符设置
QQ邮箱网页版快速登录 QQ邮箱邮箱账号官方入口地址
在J*a中如何开发简易仓库管理与库存统计_仓库管理库存统计项目实战解析
Django AJAX 文件上传教程:解决图片无法保存到模型的常见问题
为什么我的微信朋友圈看不到别人的更新_微信朋友圈更新显示异常解决方法
vivo手机互传视频怎么操作_vivo手机互传视频详细传输方法
深入理解J*aScript Promise异步执行与微任务队列
J*aScript动态修改指定div内所有a标签样式指南
夸克浏览器图书入口 夸克手机浏览器阅读入口
提升屏幕阅读器对“m”时间单位的播报准确性:HTML与CSS组合解决方案
QQ邮箱网页版邮箱入口 QQ邮箱官方登录平台
拼多多视频播放卡顿如何处理 拼多多视频播放优化技巧
抓大鹅解压小游戏 抓大鹅摸鱼解压入口
俄罗斯Yandex免登录入口_Yandex搜索引擎官网一键直达
红果短剧网页版官网入口 官方最新网址发布
php源码怎么看淘宝客系统_看php源码淘宝客系统技巧
J*aScript生成器_j*ascript异步迭代
漫蛙2漫画入口 漫蛙正版网页漫画直达网址
黑鲨3Pro怎样在相册开漫画风滤镜_iPhone黑鲨3Pro相册开漫画风滤镜【趣味滤镜】
163邮箱官方主页登录 直达网易邮箱登录核心页面
QQ邮箱登录平台入口 QQ邮箱网页版邮箱官方入口
Golang如何实现Web接口签名验证_Golang Web接口签名校验开发方法
React项目中导航栏Logo自适应布局:避免裁剪与布局溢出


2025-11-30
浏览次数:次
返回列表