将Yacc/Bison解析器与c++程序连接

Interfacing a Yacc/Bison Parser with a C++ program

本文关键字:c++ 程序 连接 Yacc Bison      更新时间:2023-10-16

这不是这个问题的重复,因为解决方案是不使用解析器!

我有一个Bison解析器,我可以运行./parser < file_to_parse。我想在我的c++程序中调用Bison解析器。我不想做的是system(./parser < file_to_parse),因为那将假设解析器是预编译的,然后我的整个程序将不是平台独立的。

如果我有一个函数:

void foo(file_name) {
   // call parser on file_name
}

那么这是怎么做到的?什么好主意吗?我认为这是调用yyparse或其他东西的情况,但我没有得到任何地方!

您使用的是适应c++的版本吗?

我用的是Alain Coetmeur的

目前,我只有方便使用flex++的代码,但它应该向您展示这是如何进行的。

当你运行flex++时,它将输出一个头文件和一个新类的实现文件来进行解析。将这些添加到项目中,然后写入:

// Construct new instance of parser
::lex * plex = new lex();
// tell parser what file to parse
plex->yyin = _wfopen(L"C:/Documents and Settings/All Users/Application Data/QuietWake/eCrew/new_rules_to_be_parsed.txt",L"r");
// run the parser
int lex_return = plex->yylex();

这正是您所做的——调用yyparse()。编写yylex()使主程序可以指定它的输入。假设您当前的yylex()调用getchar()或其他方法来读取字符;相反,您可能需要编写它来从主程序指定的某个位置获取输入。

这是C代码,但我想它也将是c++的指示性案例。
当我们必须在大学里制作编译器时,我们必须在.l文件中定义YY_DECL,像这样:

<>之前#define YY_DECL int alpha_yylex(yylval_) void *yylval_;之前

int alpha_yylex(yylval_)为函数,void *yylval为参数类型。
显然,您需要这样做,以便解析器知道从哪里开始,以及应该从哪个函数开始解析。

您还需要在您的.y文件中扩展这个函数原型(extern int alpha_yylex(void *yylval_);)。
之后,Flex/Bison实用程序将自动为您生成相关的.c文件。

一般来说,.l用于词汇规则,.y用于语法规则。因此,如果你需要从某处获取值,你必须在.y中定义一个联合,像这样(一个例子):

%union {
   union {
      int Integer;
      double Double;
      char Character;
      char* String;
   } value;
}

然后,根据你的语法和规则,你可以在你的.l file中做yylval.value.String = strdup(yytext),根据你的语法规则,在.y中做$$.Double = $1.Double + $3.Double

我不太记得了,因为我已经做了一段时间了,但我想你可以尝试一下,你可以参考文档或在这里询问:)

P.S:(我有一个很好的常见问题解答,但不记得它在哪里->如果我足够幸运找到它,我会稍后发布xD:D xD)