c - 是否可以在Flex/Lex中定义多个 'names'?

标签 c compiler-construction bison flex-lexer lex

我正在尝试使用参数 -lfl 在以下文件上运行 flex。 我收到以下错误:

romans.l:14: name defined twice
romans.l:16: name defined twice
romans.l:17: name defined twice
romans.l:19: name defined twice
romans.l:22: name defined twice
romans.l:23: name defined twice
romans.l:24: bad character: \
romans.l:24: unknown error processing section 1
romans.l:24: unknown error processing section 1
romans.l:24: bad character: {
romans.l:24: unknown error processing section 1
romans.l:24: unknown error processing section 1
romans.l:24: bad character: }
romans.l:25: unrecognised '%' directive
flex: can't open -lfl 

暂时忽略代码中的其他错误,因为我很困惑为什么第 14 行到第 23 行存在问题。据我所知,我的情况不应该有问题试图做。这是我的 lex 文件。

  1  %{
  2 // file created via echo
  3 # include <studio.h>
  4 # include "roman.tab.h"
  5 %}
  6 I{4}    { yyerror("syntax error");}
  7 V{4}    { yyerror("syntax error");}
  8 X{4}    { yyerror("syntax error");}
  9 C{4}    { yyerror("syntax error");}
 10 L{4}    { yyerror("syntax error");}
 11 D{4}    { yyerror("syntax error");}
 12 M{4}    { yyerror("syntax error");}
 13 CM      { yylval = 900; return ARABIC_NUMERAL /* NINEHUNDRED */;}
 14 M       { yylval = 1000; return ARABIC_NUMERAL /* THOUSAND */ ; }
 15 CD      { yylval = 400; return ARABIC_NUMERAL /*return FOURHUNDRED*/;}
 16 D       { yylval = 500; return ARABIC_NUMERAL /*return FIVEHUNDRED*/;}
 17 C       { yylval = 100; return ARABIC_NUMERAL /*return HUNDRED*/;}
 18 XL      { yylval = 40; return ARABIC_NUMERAL /*return FOURTY*/;}
 19 L       { yylval = 40; return ARABIC_NUMERAL /*return FIFTY*/;}
 20 IX      { yylval = 9; return ARABIC_NUMERAL /*return NINE*/; }
 21 IV      { yylval = 4; return ARABIC_NUMERAL /*return FOUR*/; }
 22 V       { yyval = 5; return ARABIC_NUMERAL /*return FIVE*/; }
 23 I       { yylval = 1; return ARABIC_NUMERAL /*return ONE*/; }
 24 \n      { return EOL }
 25 %

前面的数字只是行号(从 vim 复制并启用 :setnumber 标志)。

我假设该错误与“return ARABIC_NUMERAL”行有关,但我没有在哪里看到这明显违反了 lex 规则?真诚地在这里挠头,希望能得到任何正确方向的指点。

最佳答案

使用这个 - 在匹配规则之前和之后使用 %% 而不是 % 等。

%{
// file created via echo
#include <stdio.h>
#include "roman.tab.h"
%}
%%
I{4}    { yyerror("syntax error");}
V{4}    { yyerror("syntax error");}
X{4}    { yyerror("syntax error");}
C{4}    { yyerror("syntax error");}
L{4}    { yyerror("syntax error");}
D{4}    { yyerror("syntax error");}
M{4}    { yyerror("syntax error");}
M       { yylval = 1000; return ARABIC_NUMERAL /* THOUSAND */; }
CM      { yylval =  900; return ARABIC_NUMERAL /* NINEHUNDRED */; }
D       { yylval =  500; return ARABIC_NUMERAL /* FIVEHUNDRED */; }
CD      { yylval =  400; return ARABIC_NUMERAL /* FOURHUNDRED */; }
C       { yylval =  100; return ARABIC_NUMERAL /* HUNDRED */; }
XC      { yylval =   90; return ARABIC_NUMERAL /* NINETY */; }
L       { yylval =   50; return ARABIC_NUMERAL /* FIFTY */; }
XL      { yylval =   40; return ARABIC_NUMERAL /* FORTY */; }
X       { yylval =   10; return ARABIC_NUMERAL /* TEN */; }
IX      { yylval =    9; return ARABIC_NUMERAL /* NINE */; }
V       { yylval =    5; return ARABIC_NUMERAL /* FIVE */; }
IV      { yylval =    4; return ARABIC_NUMERAL /* FOUR */; }
I       { yylval =    1; return ARABIC_NUMERAL /* ONE */; }
\n      { return EOL; }
%%

我将 studio.h“修复”为 stdio.h,插入了缺失的 ;(在 EOL 之后) )和缺少的 l (在 yylval 中表示 V),并将第二个 40 更改为 50 (以及“四十”的拼写)。我添加了 XCX 规则 - 您似乎忽略了 90 和 10。

我还使对齐和注释更加对称;一致性在编程中非常重要。

第二个%%是可选的;第一个不是。

链接时向编译器提供 -lfl 选项;转换源代码时,您不会将其提供给 flex。这就是为什么您收到有关 -lfl 未找到的错误的原因。

关于c - 是否可以在Flex/Lex中定义多个 'names'?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53824363/

相关文章:

c - 如何将 mmap 传递给 C 中的 execve 函数?

c++ - 代码生成期间的宏替换

C++ 编译器错误 : ld: symbol(s) not found for member function referenced from _main

c++ - Yacc 中 ';' token 之前的预期类型说明符

parsing - LR(k) 到 LR(1) 语法转换

c - c 中的正则表达式 : match a digit after <n> repetitions

c - 在 C 中的文本文件中记录和替换变量

c - 为什么 8 位字段具有字节顺序?

algorithm - 如何构建具有无限循环的函数的后支配树?

linux - 我如何将 yacc 更改为 bison (LFS)