`
rstevens
  • 浏览: 95696 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

每天学点编程语言: YACC 实例分析

阅读更多
本文例子来自于 <<lex & yacc >> 第二版

LEX 负责词法分析,每次解析出一个 token。

一、 token 的类型和值
token 具有类型,在计算器例子中,包括如下类型:
1)、 NUMBER     一串数字
2)、 NAME         一个名称
3)、 '+', '-', '*', '/' 等符号

同时 token 具有值,不同类型的 token, 值的含义不一样,例如,
'1000':   类型是 NUMBER,值是1000
'abc':     类型是 NAME, 值是 'abc'

LEX 解析出一个 token 后,将此 token 的值,保存在 yylval 变量中, 并将类型返回给 YACC。
为了能保存不同类型的值, yylval  被定义成 union


%union {
    double dval;
    struct symtab *symp;
}


其中, dval 保存 NUMBER 类型的值,symp 保存 NAME 类型的值。
为了保存 NAME 类型的值,定义了一个结构


struct symtab {
    char *name;
    double (*funcptr)();
    double value;
}


其中 name 记录了“符号”的名称,而 value 则用于保存计算结果,后文再介绍。

NAME 类型的 token,又被称为 “符号”, 跟我们写程序的时候定义的变量作用相同。

因此,当 LEX 遇到数字串的时候,就把数字串的值保存到 yylval 的 dval 中,并返回 NUMBER 类型
遇到字符串的时候,根据字符串名称生成 symtab 结构,保存其名称,并将结构的地址保存到 yylval 的 symp 中,并返回 NAME 类型
遇到 '+', '-' 等符号的时候,则返回该符号的 ascii 码值。


二、 YACC 中的计算
在“产生式” 或者“规则”部分,通过 $1, $2, $3 的方式,可获取对应 token 的值,对这些变量的访问,实际就是对 yylval 的访问。此时,YACC 已经知道相应 token 的类型了,因此
对 NUMBER 类型,token 的值就是 yylval.dval
对 NAME 类型, token 的值就是  yylval.symp
例如:

statement:  NAME '=' expression { $1->value = $3;  printf("(%s) = (%g)\n", $1->name, $1->value); }<br>

expression: 
|   NUMBER          { $$ = $1;   }
|   NAME            { $$ = $1->value; }




对于 NUMBER, $1 对应的就是数值
对于 NAME, $1->name 就是符号名称

非终结符号的类型和值

1
%type <dval> expression



三、 附录: 源码


symbol.h

#define NSYMS 1024  /* maximum number of symbols */

struct symtab {
    char *name;
    double (*funcptr)();
    double value;
} symtab[NSYMS];
 
struct symtab *symlook(char* s);



 
symbol.c
 
#include "symbol.h"
#include <string.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>

struct symtab *symlook(char* s)
{
    char *p;
    struct symtab *sp;

    for(sp = symtab; sp < &symtab[NSYMS]; sp++) {
        /* is it already here? */
        if(sp->name && !strcmp(sp->name, s)) {
            printf("found symbol: (%s)\n", sp->name);
            return sp;
        }

        /* is it free */
        if(!sp->name) {
            sp->name = strdup(s);
            printf("add symbol: (%s)\n", s);
            return sp;
        }
        /* otherwise continue to next */
    }
    yyerror("Too many symbols");
    exit(1);    /* cannot continue */

} /* symlook */


  
calc.l

%{

#include "y.tab.h"
#include "symbol.h"
#include <math.h>
%}

%%

([0-9]+|([0-9]*\.[0-9]+)([eE][-+]?[0-9]+)?) {
        yylval.dval = atof(yytext);
        return NUMBER;
    }

[ \t]   ;        /* ignore white space */

[A-Za-z][A-Za-z0-9]*    {   /* return symbol pointer */
        struct symtab *sp = symlook(yytext);
        yylval.symp = sp;
        return NAME;
    }
"$" { return 0; }
\n  |
.   return yytext[0];
26
%%



 
calc.y
 
view sourceprint?
%{
#include <string.h>
#include <math.h>
#include "symbol.h"
%}

%union {
    double dval;
    struct symtab *symp;
}

%token <symp> NAME
%token <dval> NUMBER

%left '-' '+'
%left '*' '/'
%nonassoc UMINUS
%type <dval> expression
%%

statement_list: statement '\n'
    |   statement_list statement '\n'
    ;
statement:  NAME '=' expression { $1->value = $3;  printf("(%s) = (%g)\n", $1->name, $1->value); }
    |   expression      { printf("= %g\n", $1); }
    ;

expression: expression '+' expression { $$ = $1 + $3; }
    |   expression '-' expression { $$ = $1 - $3; }
    |   expression '*' expression { $$ = $1 * $3; }
    |   expression '/' expression
                {   if($3 == 0.0)
                        yyerror("divide by zero");
                    else
                        $$ = $1 / $3;
                }
    |   '-' expression %prec UMINUS { $$ = -$2; }
    |   '(' expression ')'  { $$ = $2; }
    |   NUMBER          { $$ = $1;   }
    |   NAME            { $$ = $1->value; }
    |   NAME '(' expression ')' {
            if($1->funcptr)
                $$ = ($1->funcptr)($3);
            else {
                printf("%s not a function\n", $1->name);
                $$ = 0.0;
            }
        }
    ;
%%

分享到:
评论
发表评论

文章已被作者锁定,不允许评论。

相关推荐

    Yacc.rar_Yacc java _yacc class_yacc_分析器_语法_语法分析

    "Yacc java"表明这个压缩包可能包含了将Yacc应用到Java编程环境中的实例或者库。"yacc class"可能指的是使用Java语言实现的Yacc类,用于语法分析过程。 描述中提到的“语法分析器生成工具”指的是Yacc的主要功能,...

    Lex与Yacc第二版高清版

    《Lex与Yacc第二版高清版》是一本深入解析Lex和Yacc的权威书籍,适合对编译原理、解析器构造或语言...通过深入阅读和实践,读者将能够构建自己的解析器,从而理解编程语言的内部工作机制,甚至设计和实现新的编程语言。

    Yacc.rar_yacc

    Yacc,全称为“Yet Another ...在实际应用中,YACC常被用于编程语言的编译器、解释器以及配置文件解析器等。YACC的灵活性和强大的功能使其成为编程工具箱中的重要一环,尤其是在编译原理和语言设计的学习和实践中。

    Berkeley Yacc.tar.gz

    Berkeley Yacc广泛应用于各种编程语言的编译器和解释器开发,如C、C++、Perl等。此外,在构建脚本语言解释器、配置文件解析器,甚至数据库查询解析等领域也有广泛应用。 六、学习与使用 想要掌握Berkeley Yacc,你...

    yufa.rar_LEX_fuelrdd_yacc语法分析_编译器

    "yufa.rar"包含的文件可能是作者实现的一个特定的语法分析器,它可能专注于处理某种特定的编程语言或者特定的语法结构。通过分析和运行这些文件,我们可以学习如何使用LEX和YACC来解析复杂的语言结构,了解编译器的...

    lex and yacc清晰版

    它们不仅可以处理传统的编程语言,还可以用来解析非标准格式的数据文件,或者构建自定义的命令行工具。 在学习lex和yacc时,你需要理解以下几个核心概念: 1. **正则表达式**:lex的基础,用于定义词法规则。掌握...

    Lex和Yacc从入门到精通pdf

    8. **应用实例**:通过实践,如编写小型编程语言或脚本解析器,来巩固和深化对Lex和Yacc的理解。 掌握Lex和Yacc,不仅可以提升你在编译原理领域的专业技能,还能为开发自己的语言解析工具或进行语言处理项目提供...

    lex&yacc英文第二版

    《lex&yacc英文第二版》是一本深入探讨词法分析器lex和语法分析器yacc的权威书籍,对于理解编译原理和技术有着极其重要的价值。lex和yacc是两个在软件开发领域广泛使用的工具,它们是构建解析器和编译器的核心组件,...

    LEX和YACC在软PLC开发系统中的应用

    - **语法分析器的生成**:定义软PLC编程语言的文法规则,并使用YACC生成语法分析器。这个过程涉及定义语言的语法规则,并指定如何处理不同的语法结构。 - **语法树的构造**:YACC根据文法规则分析输入的token序列,...

    lex_and_yacc.zip

    lex(也称为flex)是一个词法分析器生成器,它的主要任务是将源代码中的字符流转换为有意义的符号流,即识别出编程语言中的关键字、标识符、数字、字符串等。它通过模式匹配的方式定义规则,将符合规则的字符序列...

    lex-yacc学习文档

    《lex与yacc:深入解析编译语言构造》 在编程世界中,编译器是不可或缺的一部分,它将...通过学习和实践,你可以掌握如何从零开始构建自己的编程语言,这对于提升编程技能和理解计算机科学的核心概念具有重要意义。

    Lex和Yacc简明教程.zip_crowdywe_lex yacc_pianoxu4_yacc教程

    5. 实例分析:通过实际案例,演示 Lex 和 Yacc 在不同编程语言和工具中的应用。 6. 调试和优化:学习如何找出并修复词法和语法错误,以及如何提高分析器的性能。 通过阅读《Lex和Yacc简明教程.pdf》,无论是初学者...

    \lex&yacc

    yacc则是语法分析器生成器,它的功能是对词法分析生成的符号流进行解析,检查其是否符合程序设计语言的语法规则。yacc基于上下文无关文法(CFG)来定义语言结构,用户通过编写yacc规则文件,指定各种语法规则,yacc...

    yacc和lex词法分析开发包

    它们是软件工程中强大的工具,通过理解它们的工作原理和实践使用,开发者可以更深入地了解编程语言的本质,并能构建自己的定制解析解决方案。 总的来说,`yacc`和`lex`提供了构建语言解析器的基础框架,它们的结合...

    Lex@yacc.rar_LEX_flex_lex/flex_yacc_yacc flex

    6. 实例分析:通过一个简单的编程语言或脚本语言的词法规则示例,展示如何使用flex进行实际操作。 文件"Lex@yacc.txt"可能是对上述内容的详细阐述,而"www.pudn.com.txt"可能是从网络资源网站pudn.com下载的补充...

    yacc与lex的一个例子

    Yacc和Lex是两种强大的工具,它们在编程领域中主要用于解析语言的词法和语法。Yacc(Yet Another Compiler-Compiler)是一个基于LR语法分析的工具,它能生成解析器,处理复杂的语法结构。而Lex(也称为Flex)则是一...

    Lex和Yacc简明教程

    这些规则定义了语言的句法结构,Yacc将这些结构转换为解析树,进而进行语义分析和代码生成。Yacc生成的代码与Lex生成的词法分析器协同工作,共同构建一个完整的编译器前端。 Lex和Yacc结合使用时,首先由Lex识别出...

    msvc.zip_MSVC_yacc_yacc ms

    【标签】"msvc yacc yacc_ms" 明确了主题的关键点:与MSVC相关的YACC工具及其变种"yacc_ms"。这可能是指在MSVC下定制或优化过的YACC版本,适应Windows平台的编译需求。 【压缩包子文件的文件名称列表】中的库文件...

    Lex and Yacc

    这种组合使用方式非常高效且灵活,可以处理大多数现代编程语言。 #### 五、Lex和Yacc的应用实例 - **词法分析器示例**:例如,在处理C语言时,Lex可以通过定义如下的规则来识别关键字: ``` "int" {return INT;}...

Global site tag (gtag.js) - Google Analytics