C语言的lex和yacc工具说明

2024-04-17 18:38
文章标签 语言 工具 说明 yacc lex

本文主要是介绍C语言的lex和yacc工具说明,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Lex工具 
------- 
Lex工具是一种词法分析程序生成器,它可以根据词法规则说明书的要求来生成单词识别程序,由该程序识别出输入文本中的各个单词。 
1、lex程序的结构 

-定义部分 
-规则部分 
-用户子程序部分 

其中规则部分是必须的,定义和用户子程序部分是任选的。 

(1) 定义部分 
定义部分起始于"%{"符号,终止于"%}"符号,其间可以是包括include语句、声明语句在内的C语句。 
%{ 
#include "stdio.h" 
#include "y.tab.h" 
extern int lineno; 
%} 

(2) 规则部分 
规则部分起始于"%%"符号,终止于"%%"符号,其间则是词法规则。词法规则由模式和动作两部分组成。模式部分可以由任意的正则表达式组成,动作部分是由C语言语句组成,这些语句用来对所匹配的模式进行相应处理。需要注意的是,lex将识别出来的单词存放在yytext[]字符数据中,因此该数组的内容就代表了所识别出来的单词的内容。 

%% 
[/t] {;} 
[0-9]+/.?[0-9]*/.[0-9]+ 
{ sscanf(yytext,"%1f", &yylval.val); 
return NUMBER; } 
/n { lineno++;return ''/n''; } 
. { return yytex+[0]; } 
%% 

(3) 用户子程序部分 
用户子程序部分可以包含用C语言编写的子程序,而这些子程序可以用在前面的动作中,这样就可以达到简化编程的目的。下面是带有用户子程序的lex程序片段。 
"/*" skipcmnts(); 
. /* rest of rules */ 
%% 
skipcmnts() 
{ 
for ( ; ; ) 
{ 
while (input()!=''*''); 
if(input()!=''/'') 
unput(yytext[yylen-1]); 
else return; 


2、lex工具的使用方法 
首先编写一个lex程序 
vi lex.l 
%{ 
#include "stdio.h" 
%} 
%% 
[/n] ; 
[0-9]+ printf("Interger: %s /n",yytext); 
[0-9]*/.[0-9]+ printf("Float: %s/n",yytext); 
[a-zA-Z][a-zA-Z0-9]* printf("Word:%s/n",yytext); 
. printf("Other symbol:%c/n",yytext[0]); 
%% 

然后使用lex将lex.l转换成C语言程序 
$lex lex.l 
使用上述命令产生的C语言程序为lex.yy.c 
然后使用C编译程序将lex.yy.c编译成可执行程序regn 
$cc -c lex.yy.c 
$cc lex.yy.o -ll -o regn 
下面可以使用regn来识别单词 
$vi testfile 
x=355 
y=113 
p=x/y 

# ./regn < testfile 
Word:x 
Other symbol:= 
Interger: 355 
Word:y 
Other symbol:= 
Interger: 113 
Word:p 
Other symbol:= 
Word:x 
Other symbol:/ 
Word:y 


yacc工具 
-------- 
yacc工具是一种语法分析程序生成器,它可以将有关某种语言的语法说明书转换成相应的语法分析程序,由该程序完成对相应语言中语句的语法分析工作。 

1、yacc程序结构 
在使用yacc工具前,必须首先编写yacc程序,因为有关语法分析程序是根据yacc程序生成的。yacc程序实际上是有关语法规则的说明书,它也是由定义部分、规则部分和子程序部分组成的。yacc程序的定义部分类似于lex程序的定义部分,只是在其后可带有yacc声明,其中包括词法单词、语法变量、优先级和结合性信息。yacc程序的规则部分由语法规则和相应的动作组成,子程序部分可以包括在前面规则部分用到的子程序定义。接下来是main主程序,它调用yyparse子程序来对输入进行语法分析,而yyparse反复地调用yylex子程序来获得输入单词,在语法出错时可通过yyerror子程序来处理。 

2、yacc工具的使用方法 
实例:我们将yacc程序分成片段,把这些片段组合在一起就是yacc程序。我们要使用的语法规则是一个有关四则运算的语法规则,可用BNF范式描述 
list: expr /n 
list expr /n 
expr :NUMBER 
expr + expr 
expr - expr 
expr * expr 
expr / expr 
(expr) 
其含义是list是一个表达式序列,每个后面带有一个新行。表达式是一个数值,或是由运算符连起来的两个表达式,以及用圆括号括起来的表达式。 
下面是有关上述语法规则的yacc程序片段。 
$vi hoc.y 
%{ 
#define YYSTYPE double 
%} 
%token NUMBER 
%left ''+'' ''-'' 
%left ''*'' ''/'' 
%% 
list: 
list ''/n'' 
list expr ''/n'' { printf("/t%. 8g/n",$2);} 

expr : NUMBER {$$=$1;} 
expr ''+'' expr {$$ = $1 + $3; } 
expr ''-'' expr {$$ = $1 - $3; } 
expr ''*'' expr {$$ = $1 * $3; } 
expr ''/'' expr {$$ = $1 / $3; } 
''(''expr'')'' {$$ = $2; } 
%% 
上述yacc程序片段实际上是它的定义部分和规则部分。在yacc声明部分,%token NUMBER表明了NUMBER是一个单词符号,%left则表明了运算符号的左结合性,并且''*''和''/''和优先级比''+''和''-''的优先级高。在yacc程序的规则部分,备用规则是用''''隔开的,规则中的动作实际上是C语句序列,其中$n(即$1,$2等)是用来引用规则中的第几个成份,而$$则代表了整个规则的返回值。 

下面的yacc程序片段是main主程序 
#include <stdio.h> 
#include <ctype.h> 
char *progname; 
int lineno=1; 
main(argc,argv) 
int argc; 
char *argv[]; 
{ progname = argv[0]; 
yyparse(); 


main主程序调用yyparse子程序来处理来处理输入,而yyparse又是通过yylex子程序来获得输入单词并通过yyerror子程序来报告出错信息。下面是有关这两个子程序的yacc程序片段 

yylex() 
{ int c; 
while ((c=getchar()) == '' '' c==''/t'') ; 
if (c==EOF) 
return 0; 
if (c==''.''isdigit(c)){ 
ungetc(c,stdin); 
scanf("%lf", &yylval); 
return NUMBER; 

if(c==''/n'') 
lineno++; 
return c; 

yyerror(s) 
char *s; 
{ warning (s,(char *)0); 

warning(s,t) 
char *s,*t; 
{ fprintf(stderr,"%s:%s",progname,s); 
if(t) 
fprintf(stderr,"%s",t); 
fprintf(stderr," near line %d/n",lineno); 


这样就完成了整个yacc程序 
接下来就使用 yacc将hoc.y转换成C语言程序 
$yacc hoc.y 
使用上述命令产生的C语言程序为y.tab.c,这时可以使用C编译程序将它编译成可执行程序hoc. 
$cc y.tab.c -o hoc 

下面是使用hoc的例子 
# ./hoc 
4*3*2 
24 
(1+2)*(3+4) 
21 
1/2 
0.5 
355/133 
2.6691729 
-3-4 
./hoc:Syntax error near line 5 
上述结果显示中,分别表明了计算结果,最后一次计算出错的原因是由于在规则定义中未来定义单目减运算符号。

这篇关于C语言的lex和yacc工具说明的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/912540

相关文章

MySQL慢查询工具的使用小结

《MySQL慢查询工具的使用小结》使用MySQL的慢查询工具可以帮助开发者识别和优化性能不佳的SQL查询,本文就来介绍一下MySQL的慢查询工具,具有一定的参考价值,感兴趣的可以了解一下... 目录一、启用慢查询日志1.1 编辑mysql配置文件1.2 重启MySQL服务二、配置动态参数(可选)三、分析慢查

Go语言网络故障诊断与调试技巧

《Go语言网络故障诊断与调试技巧》在分布式系统和微服务架构的浪潮中,网络编程成为系统性能和可靠性的核心支柱,从高并发的API服务到实时通信应用,网络的稳定性直接影响用户体验,本文面向熟悉Go基本语法和... 目录1. 引言2. Go 语言网络编程的优势与特色2.1 简洁高效的标准库2.2 强大的并发模型2.

Java中数组与栈和堆之间的关系说明

《Java中数组与栈和堆之间的关系说明》文章讲解了Java数组的初始化方式、内存存储机制、引用传递特性及遍历、排序、拷贝技巧,强调引用数据类型方法调用时形参可能修改实参,但需注意引用指向单一对象的特性... 目录Java中数组与栈和堆的关系遍历数组接下来是一些编程小技巧总结Java中数组与栈和堆的关系关于

Go语言使用sync.Mutex实现资源加锁

《Go语言使用sync.Mutex实现资源加锁》数据共享是一把双刃剑,Go语言为我们提供了sync.Mutex,一种最基础也是最常用的加锁方式,用于保证在任意时刻只有一个goroutine能访问共享... 目录一、什么是 Mutex二、为什么需要加锁三、实战案例:并发安全的计数器1. 未加锁示例(存在竞态)

C语言自定义类型之联合和枚举解读

《C语言自定义类型之联合和枚举解读》联合体共享内存,大小由最大成员决定,遵循对齐规则;枚举类型列举可能值,提升可读性和类型安全性,两者在C语言中用于优化内存和程序效率... 目录一、联合体1.1 联合体类型的声明1.2 联合体的特点1.2.1 特点11.2.2 特点21.2.3 特点31.3 联合体的大小1

基于Python实现进阶版PDF合并/拆分工具

《基于Python实现进阶版PDF合并/拆分工具》在数字化时代,PDF文件已成为日常工作和学习中不可或缺的一部分,本文将详细介绍一款简单易用的PDF工具,帮助用户轻松完成PDF文件的合并与拆分操作... 目录工具概述环境准备界面说明合并PDF文件拆分PDF文件高级技巧常见问题完整源代码总结在数字化时代,PD

Python按照24个实用大方向精选的上千种工具库汇总整理

《Python按照24个实用大方向精选的上千种工具库汇总整理》本文整理了Python生态中近千个库,涵盖数据处理、图像处理、网络开发、Web框架、人工智能、科学计算、GUI工具、测试框架、环境管理等多... 目录1、数据处理文本处理特殊文本处理html/XML 解析文件处理配置文件处理文档相关日志管理日期和

使用Python开发一个Ditto剪贴板数据导出工具

《使用Python开发一个Ditto剪贴板数据导出工具》在日常工作中,我们经常需要处理大量的剪贴板数据,下面将介绍如何使用Python的wxPython库开发一个图形化工具,实现从Ditto数据库中读... 目录前言运行结果项目需求分析技术选型核心功能实现1. Ditto数据库结构分析2. 数据库自动定位3

Go语言使用select监听多个channel的示例详解

《Go语言使用select监听多个channel的示例详解》本文将聚焦Go并发中的一个强力工具,select,这篇文章将通过实际案例学习如何优雅地监听多个Channel,实现多任务处理、超时控制和非阻... 目录一、前言:为什么要使用select二、实战目标三、案例代码:监听两个任务结果和超时四、运行示例五

C语言中%zu的用法解读

《C语言中%zu的用法解读》size_t是无符号整数类型,用于表示对象大小或内存操作结果,%zu是C99标准中专为size_t设计的printf占位符,避免因类型不匹配导致错误,使用%u或%d可能引发... 目录size_t 类型与 %zu 占位符%zu 的用途替代占位符的风险兼容性说明其他相关占位符验证示