答案:文章介绍了在C++中构建简单抽象语法树(AST)的过程,涵盖节点基类定义、具体节点类型实现、变量环境管理、词法分析器与递归下降解析器的设计,并通过示例展示表达式解析与求值流程。
在C++中实现一个简单的抽象语法树(AST)是理解编译器或解释器工作原理的关键一步。AST 是源代码结构的树形表示,它忽略掉源码中的语法细节(如括号、分号),专注于程序的逻辑结构。下面我们将一步步构建一个极简的 AST,并结合词法分析和语法分析来解析简单表达式。
所有AST节点都应继承自一个公共基类。我们使用多态来处理不同类型的节点,比如数字、变量、二元操作等。
#include#include #include
根据常见表达式元素,定义几种基本节点:
// 数字常量节点
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 zuojiankuohaophpcnzuojiank
uohaophpcn "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
# 求值
# 加减
# 构建一个
# 加减乘除
# 你可以
# 第一个
# 我们可以
# 几种
# 环境管理
相关文章:
怎么将XML数据可视化 D3.js加载XML
如何在IIS管理器中快速创建并配置网站?
山东云建站价格为何差异显著?
东莞专业网站制作公司有哪些,东莞招聘网站哪个好?
,石家庄四十八中学官网?
如何快速启动建站代理加盟业务?
网站制作的方法有哪些,如何将自己制作的网站发布到网上?
制作网站软件推荐手机版,如何制作属于自己的手机网站app应用?
安徽网站建设与外贸建站服务专业定制方案
网站制作免费,什么网站能看正片电影?
如何优化Golang Web性能_Golang HTTP服务器性能提升方法
大连网站制作费用,大连新青年网站,五年四班里的视频怎样下载啊?
如何快速搭建个人网站并优化SEO?
如何选择长沙网站建站模板?H5响应式与品牌定制哪个更优?
大连 网站制作,大连天途有线官网?
手机钓鱼网站怎么制作视频,怎样拦截钓鱼网站。怎么办?
网站制作中优化长尾关键字挖掘的技巧,建一个视频网站需要多少钱?
外贸公司网站制作,外贸网站建设一般有哪些步骤?
外汇网站制作流程,如何在工商银行网站上做外汇买卖?
PHP正则匹配日期和时间(时间戳转换)的实例代码
网站好制作吗知乎,网站开发好学吗?有什么技巧?
c# Task.ConfigureAwait(true) 在什么场景下是必须的
建站之星导航如何优化提升用户体验?
建站之星如何配置系统实现高效建站?
如何快速搭建高效简练网站?
南京网站制作费用,南京远驱官方网站?
Python多线程使用规范_线程安全解析【教程】
巅云智能建站系统:可视化拖拽+多端适配+免费模板一键生成
小程序网站制作需要准备什么资料,如何制作小程序?
成都品牌网站制作公司,成都营业执照年报网上怎么办理?
成都网站制作报价公司,成都工业用气开户费用?
如何用好域名打造高点击率的自主建站?
定制建站方案优化指南:企业官网开发与建站费用解析
小捣蛋自助建站系统:数据分析与安全设置双核驱动网站优化
天津个人网站制作公司,天津网约车驾驶员从业资格证官网?
购物网站制作公司有哪些,哪个购物网站比较好?
Python路径拼接规范_跨平台处理说明【指导】
seo网站制作优化,网站SEO优化步骤有哪些?
如何自定义建站之星网站的导航菜单样式?
建站之星代理平台如何选择最佳方案?
建站之星如何防范黑客攻击与数据泄露?
自助网站制作软件,个人如何自助建网站?
如何通过VPS建站无需域名直接访问?
建站之星展会模版如何一键下载生成?
广州网站设计制作一条龙,广州巨网网络科技有限公司是干什么的?
高性价比服务器租赁——企业级配置与24小时运维服务
如何正确选择百度移动适配建站域名?
建站主机与虚拟主机有何区别?如何选择最优方案?
如何在阿里云通过域名搭建网站?
如何用狗爹虚拟主机快速搭建网站?
*请认真填写需求信息,我们会在24小时内与您取得联系。