我正在尝试用Bison构建一个解析器。我在第一部分中有以下内容:
%union {
int ttype;
// enums used in lexer
Staff stafftype;
Numeral numeral;
Quality quality;
Inversion inversion;
Pitch pitch;
Accidental accidental;
// Classes used in parser
Roman roman;
}
%token <stafftype> STAFFTYPE
%token <numeral> NUMERAL
%token <quality> QUALITY
%token <inversion> INVERSION
%token <pitch> PITCH
%token <accidental> ACCIDENTAL
%token <ttype> COLON
%token <ttype> SLASH
%token <ttype> COMMA
%type <roman> accidentalRoman
带有一些语法规则。这是一个:
accidentalRoman
: NUMERAL { $$ = Roman($1); }
| ACCIDENTAL NUMERAL { $$ = Roman($2, $1); }
;
我基本上有三个相关的问题。
%union真正代表什么?我认为它代表词法分析器可以返回的类型。我的词法分析器规则包含类似
return STAFFTYPE
的语句,以指示我已在yylval.stafftype
中填充了Staff
对象。很公平。然而;联合似乎也与语法动作中的
$$ =
语句有关。为什么语法动作的结果类型需要合并在一起?在我的示例中,
Roman
类具有带参数的构造函数。但是,联合中的声明导致错误no matching function for call to 'Roman::Roman()'
。有没有办法解决?我正在尝试使用$$ =
构建解析树,并且树中的节点肯定在其构造函数中需要参数。实际上,它甚至不允许使用0参数的构造函数:error: union member 'YYSTYPE::roman' with non-trivial 'Roman::Roman().
最佳答案
%union
真正代表什么?我认为它代表词法分析器可以返回的类型。
否。它表示生产可以通过$$ =
返回的类型。词法分析器仅返回通过%token
指令定义的整数常量。该词法分析器可以填充yylval
成员作为副作用,但是从任何意义上讲,它都不是该词法分析器的返回类型。
我的词法分析器规则包含诸如return STAFFTYPE之类的语句,以指示我已使用Staff对象填充yylval.stafftype。
他们不应该。它们应该返回语法中使用的标记类型,并且通常不应该将任何内容放入yylval
中,除非是文字。您正在使用语法分析器应该做的词法分析器工作。
工会似乎也与语法动作中的$$ = statements
有关。为什么语法动作的结果类型需要合并在一起?
因为那是放置它们的地方。在yylval
值堆栈的顶部。
在我的示例中,罗马类具有带有参数的构造函数。但是,联合中的声明会导致没有错误的匹配函数调用'Roman :: Roman()'。有没有办法解决?我正在尝试使用$$ =
建立一个解析树,树中的节点肯定在其构造函数中需要参数。实际上,它甚至不允许使用0参数的构造函数:错误:联合成员YYSTYPE::roman
与平凡的Roman::Roman()
。
通常,%union
应该由整数,双精度型,其他原始类型和指针组成。无论如何,联合中的对象都是有问题的,并且在解析器堆栈上大部分都是空间的浪费。
关于c++ - 用Bison解析:构造函数的作用,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/45727327/