基于flex&goyacc实现的计算器

栏目: C · 发布时间: 5年前

内容简介:特性简介:

基于flex&goyacc实现的计算器

基于flex&goyacc实现的计算器

特性简介:

  • 支持整数四则运算

  • 支持小括弧提升优先级

  • 支持临时变量保存结果

安装和使用(需要有GCC环境):

$ go get github.com/chai2010/calculator
$ calculator
1+2*3
= 7
x=3-(2-1)
= 2
x*2
= 4

词法符号

先创建 tok.h 文件,包含词法符号:

enum {
    ILLEGAL = 10000,
    EOL = 10001,

    ID = 258,
    NUMBER = 259,

    ADD = 260, // +
    SUB = 261, // -
    MUL = 262, // *
    DIV = 263, // /
    ABS = 264, // |

    LPAREN = 265, // (
    RPAREN = 266, // )
    ASSIGN = 267, // =
};

其中 ILLEGAL 表示不能识别的无效的符号, EOL 表示行的结尾,其它的符号也字面含义相同。

词法解析

然后创建 calc.l 文件,定义每种词法的正则表达式:

%option noyywrap

%{
#include "tok.h"
%}

%%

[_a-zA-Z]+ { return ID; }
[0-9]+     { return NUMBER; }

"+"    { return ADD; }
"-"    { return SUB; }
"*"    { return MUL; }
"/"    { return DIV; }
"|"    { return ABS; }

"("    { return LPAREN; }
")"    { return RPAREN; }
"="    { return ASSIGN; }

\n     { return EOL; }
[ \t]  { /* ignore whitespace */ }
.      { return ILLEGAL; }

%%

最开始的 noyywrap 选项表示关闭 yywrap 特性,也就是去掉对flex库的依赖,生成可移植的词法分析器代码。然后在 %{%} 中间是原生的 C语言 代码,通过包含 tok.h 引入了每种记号对应的枚举类型。在两组 %% 中间的部分是每种记号对应的正则表达式,先出现的优先匹配,如果匹配失败则继续尝试后面的规则。每个正则表达式后面跟着一组动作代码,也就是普通的C语言代码,这里都是返回记号的类型。

然后通过flex工具生成C语言词法解析器文件:

$ flex --prefix=yy --header-file=calc.lex.h -o calc.lex.c calc.l

其中 --prefix 表示生成的代码中标识符都是以 yy 前缀。在一个项目有多个flex生成代码时,可通过前缀区分。 --header-file 表示生成头问题,这样方便在其它代码中引用生成的词法分析函数。 -o 指定输出源代码文件的名字。

生成的词法分析器中,最重要的有以下几个:

extern int yylineno;
extern char *yytext;

extern int yylex (void);

其中 yylineno 表示当前的行号, yytext 表示当前记号对应的字符串。而 yylex 函数每次从标准输入读取一个记号,返回记号类型的值(在 tok.h 文件定义),如果遇到文件结尾则返回0。

如果需要从字符串解析,则需使用以下的导出函数:

YY_BUFFER_STATE yy_scan_bytes (yyconst char *bytes,yy_size_t len  );

通过 yy_scan_bytes 函数,可以设置字符串作为要解析的目标,然后每次调用 yylex 函数就会从字符串读取数据。这些函数都在 calc.lex.h 文件中声明。

将C语言词法分析器包装为 Go 函数

创建 lex.go 文件,内容如下:

package main

//#include "tok.h"
//#include "calc.lex.h"
import "C"

type calcLex struct {}

func newCalcLexer(data []byte) *calcLex {
    p := new(calcLex)
    C.yy_scan_bytes((*C.char)(C.CBytes(data)), C.yy_size_t(len(data)))
    return p
}

func (p *calcLex) Lex(yylval *calcSymType) int {
    var tok = C.yylex()
    var yylineno = int(C.yylineno)
    var yytext = C.GoString(C.yytext)

    switch tok {
    case C.ID:
        // yylval.id = yytext
        return ID

    case C.NUMBER:
        //yylval.value, _ = strconv.Atoi(yytext)
        return NUMBER

    case C.ADD:
        return ADD
    // ...

    case C.EOL:
        return EOL
    }

    if tok == C.ILLEGAL {
        log.Printf("lex: ILLEGAL token, yytext = %q, yylineno = %d", yytext, yylineno)
    }

    return 0 // eof
}

新建的 calcLex 类型对应Go语言版本的词法分析器,底层工作通过CGO调用flex生成的C语言函数完成。首先 newCalcLexer 创建一个词法分析器,参数是要分析的数据,通过 C.yy_scan_bytes 函数调用表示从字符串解析记号。然后 calcLex 类型的 Lex 方法表示每次需要解析一个记号(暂时忽略方法的 calcSymType 参数),内部通过调用 C.yylex() 读取一个记号,同时记录行号和记号对应的字符串。最后将C语言的记号转为Go语言的记号值返回,比如 C.ID 对应Go语言的 ID

对应 ID 类型, yytext 表示变量的名字。对于 NUMBER 类型, yytext 保护数字对应的字符串,可以从字符串解析出数值。但是,Go语言的词法分析器如何返回变量的名字或者是数字的值呢?答案是通过 Lex*calcSymType 类型的参数可以记录记号额外的属性值。而 calcSymType 类型是由 goyacc 工具生成的代码,在下面我们将介绍yacc的内容。

`goyacc`生成语法解析器

goyacc 是Go语言版本的yacc工具,是由Go语言官方团队维护的扩展包工具。

创建 calc.y 文件:

%{
package main

var idValueMap = map[string]int{}
%}

%union {
    value int
    id    string
}

%type  <value> exp factor term
%token <value> NUMBER
%token <id>    ID

%token ADD SUB MUL DIV ABS
%token LPAREN RPAREN ASSIGN
%token EOL

%%
calclist
    : // nothing
    | calclist exp EOL {
        idValueMap["_"] = $2
        fmt.Printf("= %v\n", $2)
    }
    | calclist ID ASSIGN exp EOL {
        idValueMap["_"] = $4
        idValueMap[$2] = $4
        fmt.Printf("= %v\n", $4)
    }
    ;

exp
    : factor         { $$ = $1 }
    | exp ADD factor { $$ = $1 + $3 }
    | exp SUB factor { $$ = $1 - $3 }
    ;

factor
    : term            { $$ = $1 }
    | factor MUL term { $$ = $1 * $3 }
    | factor DIV term { $$ = $1 / $3 }
    ;

term
    : NUMBER            { $$ = $1 }
    | ID                { $$ = idValueMap[$1] }
    ;

%%

和flex工具类型,首先在 %{%} 中间是原生的Go语言代码。然后 %union 定义了属性值,用于记录语法解析中每个规则额外的属性值。通过 %type 定义BNF规则中非终结的名字, %token 定义终结记号名字(和flex定义的记号类型是一致的)。而 %type%token 就可以通过 <value><id> 的可选语法,将后面的名字绑定到属性。就是后续代码中 $$ 对应的属性,比如 %token <id> ID 表示 ID 对应的属性为 id ,因此在后面的 ID { $$ = idValueMap[$1] } 表示数值 id 属性的值,其中 idValueMap 用于管理变量的值。

然后通过goyacc工具生成代码:

$ goyacc -o calc.y.go -p "calc" calc.y

其中 -o 指定输出的文件名, -p 指定标识符名字前缀(和flex的 --prefix 用法类似)。在生成的 calc.y.go 文件中将包含最重要的 calcParse 函数,该函数从指定的词法解析器中读取词法,然后进行语法分析。同时将包含 calcSymType 类型的定义,它是 Lex 词法函数的输出参数的类型。

在绑定了属性之后,还需要继续完善 Lex 词法函数的代码:

func (p *calcLex) Lex(yylval *calcSymType) int {
    var tok = C.yylex()
    var yylineno = int(C.yylineno)
    var yytext = C.GoString(C.yytext)

    switch tok {
    case C.ID:
        yylval.id = yytext
        return ID

    case C.NUMBER:
        yylval.value, _ = strconv.Atoi(yytext)
        return NUMBER

    ...
}

其中 yylval.id = yytext 表示词法将解析得到的变量名字填充到 id 属性中。而数字部分则是通过 yylval.value 属性保存。

运行计算器

创建main函数:

func main() {
    calcParse(newCalcLexer([]byte("1+2*3")))
}

newCalcLexer 构造一个词法解析器,然后 calcParse 语法解析器将从词法解析器依次读取记号并解析语法,在解析语法的同时将进行表达式求值运算,同时更新 idValueMap 全局的变量。

基于flex&goyacc实现的计算器

基于flex&goyacc实现的计算器


以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

风向

风向

何宝宏 / 人民邮电出版社 / 2019-1 / ¥68.00元

★这是处于不断变化的互联网时代,行业从业者与非专业从业者都应阅读的解惑之书。 ★揭示互联网思想和精神的“内核”,帮助更多人了解互联网基因。 ★看清人工智能、区块链、大数据、云计算等技术发展的规律和机会。 ★为投资者、创业者提供方向,为广大技术从业者了解技术,为就业择业者提供建议和参考。 ★中国信通院院长刘多、腾讯云总裁邱跃鹏做序推荐。 ★中国工程院院士邬贺铨、中国科学......一起来看看 《风向》 这本书的介绍吧!

RGB转16进制工具
RGB转16进制工具

RGB HEX 互转工具

XML 在线格式化
XML 在线格式化

在线 XML 格式化压缩工具

Markdown 在线编辑器
Markdown 在线编辑器

Markdown 在线编辑器