Bison|4.1 解析函数 yyparse|V20240916

声明:本文梳理自 Bison 参考手册:https://www.gnu.org/software/bison/manual/bison.html;部分借鉴自通义千问 AI。

4.1 解析函数 yyparse

你调用函数 yyparse 来触发解析过程。这个函数读取标记(tokens),执行动作,并最终在遇到输入结束或无法恢复的语法错误时返回。你也可以编写一个动作来指示 yyparse 立即返回,不再进一步读取。

函数int yyparse (void):如果解析成功(即返回是由于输入结束),则 yyparse() 函数的返回值是 0。如果解析因为无效输入而失败,即包含语法错误的输入或导致 YYABORT 被调用的情况,则返回值是 1。如果解析由于内存耗尽而失败,则返回值是 2。

在语义组行为(action)中,可以使用以下宏来使 yyparse 立即返回:

YYACCEPT:立即返回 0(解析成功)

YYABORT:立即返回 1(因无效输入而失败)

YYNOMEM:立即返回 2(因内存不足而失败)

如果你使用了一个可重入的解析器,可以选择以可重入的方式向其传递额外的参数信息。要做到这一点,可以使用声明 %parse-param

声明%parse-param {argument-declaration} ...:声明一个或多个参数声明为附加的 yyparse() 函数参数。参数声明在声明函数或原型时使用。参数声明中的最后一个标识符必须是参数名。

这是一个示例。在解析器中这样写:

%parse-param {int *nastiness} {int *randomness}

然后像这样调用解析器:

{
  int nastiness, randomness;/* 将适当的数据存入 nastiness 和 randomness 中。*/
  value = yyparse (&nastiness, &randomness);}

在语义组行为(action)中,可以使用如下表达式来引用这些数据:

exp:{; *randomness += 1;}

使用下面的声明:

%parse-param {int *randomness}

将会得到如下的函数签名:

void yyerror (int *randomness, const char *msg);
int  yyparse (int *randomness);

或者,如果同时使用了 %define api.pure full (或仅 %define api.pure)和 %locations

void yyerror (YYLTYPE *llocp, int *randomness, const char *msg);
int  yyparse (int *randomness);

这里的 YYLTYPE 是用于存储位置信息的数据类型。

作者: 胡彦 本框架是一个lex/yacc完整的示例,用于学习lex/yacc程序基本的搭建方法,在linux/cygwin下敲入make就可以编译和执行。 本例子虽小却演示了lex/yacc程序最常见和重要的特征: * lex/yacc文件格式、程序结构。 * 如何在lex/yacc中使用C++和STL库,用extern "C"声明那些lex/yacc生成的、要链接的C函数,如yylex(), yywrap(), yyerror()。 * 重定义YYSTYPE/yylval为复杂类型。 * 用%token方式声明yacc记号。 * 用%type方式声明非终结符的类型。 * lex里正则表达式的定义、识别方式。 * lex里用yylval向yacc返回属性值。 * 在yacc嵌入的C代码动作里,对记号属性($1, $2等)、和非终结符属性($$)的正确引用方法。 * 对yyin/yyout重赋值,以改变yacc默认的输入/输出目标。 * 如何开始解析(yyparse函数),结束或继续解析(yywrap函数)。 本例子功能是,对当前目录下的file.txt文件,解析出其中的标识符、数字、其它符号,显示在屏幕上。linux调试环境是Ubuntu 10.04。 总之,大部分框架已经搭好了,你只要稍加扩展就可以成为一个计算器之类的程序,用于《编译原理》的课程设计。 文件列表: lex.l: lex程序文件。 yacc.y: yacc程序文件。 main.hpp: 共同使用的头文件。 Makefile: makefile文件。 file.txt: 给程序解析的文本文件。 使用方法: 1-把lex_yacc_example.rar解压到linux/cygwin下。 2-命令行进入lex_yacc_example目录。 3-敲入make,这时会自动执行以下操作: (1) 自动调用flex编译.l文件,生成lex.yy.c文件。 (2) 自动调用bison编译.y文件,生成yacc.tab.c和yacc.tab.h文件。 (3) 自动调用g++编译、链接出可执行文件main。 (4) 自动执行main,得到如下结果:。 bison -d yacc.y g++ -c lex.yy.c g++ -c yacc.tab.c g++ lex.yy.o yacc.tab.o -o main id: abc id: defghi int: 123 int: 45678 op: ! op: @ op: # op: $ AllId: abc defghi 参考资料:《Lex和Yacc从入门到精通(6)-解析C-C++包含文件》, http://blog.csdn.net/pandaxcl/article/details/1321552 其它文章和代码请留意我的blog: http://blog.csdn.net/huyansoft 2013-4-27
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

长行

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值