未定义对`yylex'的引用

问题描述 投票:0回答:4

我正在尝试使用 flex 和 bison 解析一个输入文件,但在编译程序时遇到一个困难。我附上了我的 Flex 和 Bison 代码以及我收到的错误。

请帮我解决这些错误

lex.l

%{
 #include <iostream>  
 #include <stdio.h>
#include "yacc.tab.h"
 #define YY_DECL extern "C" int yylex()

using namespace std;  


%}

DOT             "."  
COLON           ":"  
SEMICOLON       ";"  
COMMA           ","  
ANGLE_LEFT      "<"  
ANGLE_RIGHT     ">"  
AT              "@"  
EQUAL           "="  
SQUARE_OPEN     "["  
SQUARE_CLOSE    [^\\]"]"  
OPENBRACE       "\("  
CLOSEBRACE      "\)"  
QUOTE           "\""  
QUOTE_OPEN      "\""  
QUOTE_CLOSE     [^\\]"\""  
SPACE           " "  
TAB             "\t"  
CRLF            "\r\n"  
QUOTED_PAIR     "\\"[^\r\n]  
DIGIT           [0-9]  
ALPHA           [a-zA-Z]  
QTEXT           [0-9a-zA-Z!#$%&'()*+,\-.\/:;<=>?@\[\]^_`{|}~]  

%%

[a-zA-Z0-9]+            { yylval.sval = strdup(yytext); return TOK_STRING; }

{SPACE}*                    {return TOK_SPACE; }

{SPACE}*Name.*              {return TOK_NAME; }  
{SPACE}*SIZE.*              {return TOK_SIZE; }  
{SPACE}*ITERATE.*           {return TOK_ITERATE; }  
{SPACE}*DIRECTION.*     {return TOK_DIRECTION; }  

^{CRLF}                         { return TOK_EMPTY_LINE; }  
{CRLF}                          {}  
.                               {}/* ignore unknown chars */  

yacc.y

 %{
 #include <cstdio> 
 #include <cstring>
 #include <iostream>
 #include <stdio.h>

using namespace std;

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

void yyerror(const char* s);  


%}

%union  
{  
    char* sval;  
};  

%token <sval> TOK_NAME  
%token <sval> TOK_SIZE  
%token <sval> TOK_STRING  
%token <sval> TOK_ITERATE  
%token <sval> TOK_DIRECTION  


%token TOK_SPACE  


%%

str:  
    TOK_SPACE TOK_NAME TOK_SPACE TOK_STRING   
    {  
        cout << "Value:" << $2 << "->" << $4;  
    }  
    ;  
%%  

int main(void) {  
    FILE * pt = fopen("new file ", "r" );  
    if(!pt)  
    {  
    cout << "Bad Input.Noexistant file" << endl;  
    return -1;  
    }  
    yyin = pt;  
    do  
    {  
        yyparse();  
     }while (!feof(yyin));        
}  
void yyerror(const char *s)  
{  
   cout << "Error. " << s << endl;   
   exit(-1);     
}  

我使用以下方法构建这些:

flex bas.l 
bison -d yacc.y 
g++ lex.yy.c yacc.tab.c -lfl -o scanner.exe 

编译程序时发现以下错误:

/tmp/cceIyDkD.o:在函数“main”中:
yacc.tab.c:(.text+0x708): `main' 的多重定义
/tmp/ccatq95p.o:lex.yy.c:(.text+0x1228):首先在这里定义
/usr/bin/ld:警告:符号“main”的大小从 /tmp/ccatq95p.o 中的 86 更改为 /tmp/cceIyDkD.o 中的 120
/tmp/cceIyDkD.o:在函数 `yyparse()' 中:
yacc.tab.c:(.text+0x2d6): 对 `yylex' 的未定义引用
collect2: ld 返回 1 退出状态

请帮帮我

c++ c bison flex-lexer
4个回答
8
投票

我遇到了类似的问题,事实证明 flex 2.6.x 改变了 yylex 的行为方式。我通过使用旧版本的 Flex 解决了这个问题。在 Ubuntu 中,有一个名为 flex-old 的软件包,它是旧版本。用 flex-old 替换 flex 解决了我的这个问题。


3
投票
  1. 在 parser.y 中,更准确地说是包含内容的代码的 C 部分,您需要编写:

    extern int yylex(void);
    。不需要 Extern 关键字,因为默认情况下全局函数具有动态连接,但最好编写关键字来指出该函数不是来自您正在工作的文件。但是,您的代码在这里没有错误。

  2. 将 lex.yy.c 编译为目标文件,将 y.tab.c 编译为目标文件,然后在最终程序中将它们链接在一起。 首先执行

    yacc
    命令非常重要,因为 Lex 使用的是由 Yacc 生成的 yacc.tab.h。 你会收到错误,因为你弯曲它,但你不知道 yacc.tab.h 是什么.

    yacc -d yacc.y
    制作 y.tab.c

    g++ -c y.tab.c -o y.tab.o
    将 y.tab.c 编译为目标文件 y.tab.o

    flex lex.l
    制作 lex.yy.c

    g++ -c lex.yy.c -o lex.yy.o
    将 lex.yy.c 编译为目标文件 lex.yy.o

    g++ lex.yy.o y.tab.o -o program
    将它们链接在一起。

注意: 确保在您的 lex 文件中,在其他包含之后包含“y.tab.h”,因为其他包含可能使用 y.tab.h 中声明的某些函数,因此请记住这一点。当您编写更复杂的解析器时,可能会发生这种情况。


2
投票

代码中存在一些错误,导致其无法编译。您需要添加到文件 yacc.y:

%token <sval> TOK_EMPTY_LINE 

最后:

#include "lex.yy.c"

并以这种方式构建:

flex bas.l 
bison -d yacc.y 
g++ yacc.tab.c -lfl -o scanner.exe 

你会发现它现在可以编译了。

您可能需要注意,我删除了问题中词法分析器源中的几个拼写错误。并非所有规则都从第一列开始。我删除了几个前导空格,因为它们是拼写错误。


0
投票

我在尝试使用 C++ 编译 Flex + Bison 程序时遇到了同样的问题,但不在 C++ 模式下使用它们(这是它自己的另一个令人头疼的问题)。

在 Flex 生成的源代码、

yy.c
文件或类似文件中查找此行:

extern int yylex (void);

并将其替换为

extern "C" int yylex (void);

这个 hack 为我“解决”了它,但应该有更好的解决方案。

© www.soinside.com 2019 - 2024. All rights reserved.