gpt4 book ai didi

c++ - Flex & Bison C++

转载 作者:行者123 更新时间:2023-12-03 07:02:02 24 4
gpt4 key购买 nike

我想启动一个 Flex & Bison 翻译器,它将从给定的文件中读取并输出到另一个文件,同时理解我们在输入中给他的内容。
例如,如果我给 "This is a string" 12 4.5 ,输出文件将是
String > "这是一个字符串"
空间 > *
整数 > 12
空间 > *
float > 4.5
问题是我试图在所有这些下面的基础下,我从读取输入文件和输出文件并打开它们的地方开始。我正在处理 Visual Studio 所以我将命令行参数添加为 read exe input.txt output.txt我在 Grammar.y 的主要类(class)中管理它们文件。之后,我尝试使用 yylex(); 时返回的内容做一些正则表达式。功能找到了一些东西。我正在提供代码,但到目前为止我有 2 个问题。
首先,编译器吐出我没有声明标记,但我已将它们放在 .y 中。我在下面提供的文件。
其次,我填充就像我不知道我做错了什么,它根本不起作用,所以任何建议都会有所帮助。
这是我的Grammar.l文件

%option noyywrap

%{
#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include "Grammar.tab.h"
#define YY_DECL int yylex(yy::parser::semantic_type *yylval)
FILE *fout;

using namespace std;
%}

%%

[0-9]+ { yylval->ival = atoi(yytext); return INTEGER; }
[0-9]+"."[0-9]+ | "."?[0-9]? { yylval->fval = atof(yytext); return FLOAT; }
[a-zA-Z0-9]+ { yylval->sval = yytext; return STRING; }
" "* { return SPACE; }
"\t"* { return TAB; }

%%
这是 Grammar.y文件
%language "C++"
%start root

%{
#include <stdio.h>
#include <stdlib.h>
#include "Grammar.tab.h"
#define SIZE 512

using namespace std;

extern "C" int yylex();
extern "C" FILE *yyin;
extern "C" FILE *fout;

extern int yylex(yy::parser::semantic_type *yylval);
%}

%union{
int ival;
float fval;
char* sval;
}

%token <ival> INTEGER
%token <fval> FLOAT
%token <sval> STRING
%token SPACE
%token TAB

%%

root : ;

%%

void main( int argc, char ** argv){

if ( argc < 4 ){
printf("\nError!!! Missing Command line arguments\nUsage exe <inputfile> <outputfile>");
exit(1);
}
else{
fopen_s(&yyin, argv[3],"r");
if (yyin == NULL) {
printf("\033[0;31mError oppening input file.\033[0m");
}

fopen_s(&fout, argv[4],"r");
if (fout == NULL) {
printf("\033[0;31mError oppening output file.\033[0m");
}

do
{
yylex();
}while(!feof(yyin));
fclose(yyin);
}
fclose(fout);
}

namespace yy{
void parser::error (const location_type& loc, const std::string& msg){
std::cerr << "error at " << loc << ": " << msg << std::endl;
}
}

最佳答案

当您请求 C++ 解析器时,bison 会将 token 类型保留在全局命名空间之外。这使得用法与您在 Internet 上找到的大多数示例完全不同,这些示例假定 C 接口(interface)。
所以不要只使用 INTEGER ,例如,您需要指定全名:

[0-9]+                          { yylval->ival = atoi(yytext);
return yy::parser::token::INTEGER; }
你可以用 using 缩短一点。序言中的指示。
你的编译器也会提示对 yylex 的调用。在您的 main 内功能。请注意,您已声明 yylex作为:
extern int yylex(yy::parser::semantic_type *yylval);
这意味着它需要一个参数,它是指向 yy::parser::semantic_type 的指针。 (即 union 由您的 %union 声明所描述)。那么,为了调用该函数,您需要一个指向此类对象的指针,这意味着您需要该对象的一个​​实例来指向:
    yy::parser::semantic_type yylval;
int token;
do
{
token = yylex(&yylval);
/* Here you need to do something in order to print
* the token and possibly the associated semantic value.
* So you'll probably need something like this:
*/
switch(token) {
case yy::parser::token::INTEGER {
fprintf(fout, "INTEGER: %d\n", yylval->ival);
break;
}
// ...
case 0: break;
}
} while(token);
请注意,我更改了 feof测试以便循环在 yylex 时终止返回 0,这就是 yylex表示它已到达输入结束。

关于c++ - Flex & Bison C++,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64696972/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com