LL(1)文法判断程序

基于 java 编写,编译原理 LL(1)文法的简单判断LL(1)文法的描述1、一个上下文无关文法是LL(1)文法的充分必要条件是:对每个非终结符A的两个不同产生式,A→α, A→β,满足SELECT(A→α)∩SELECT(A→β)= Φ( 其中α,β不同时能推导出ε)。2、LL(1)文法的含义:   第一个L即从左到右扫描输入串   第二个L即生成的是最左推导向右看一个输入...
摘要由CSDN通过智能技术生成

基于 java 编写,编译原理 LL(1)文法的简单判断

LL(1)文法的描述

1、一个上下文无关文法是LL(1)文法的充分必要条件是:对每个非终结符A的两个不同产生式,A→α, A→β,满足SELECT(A→α)∩SELECT(A→β)= Φ( 其中α,β不同时能推导出ε)。

2、LL(1)文法的含义:

   第一个L即从左到右扫描输入串

   第二个L即生成的是最左推导向右看一个输入符号便可决定选择哪个产生式。

3、LL(1)文法的判别:当我们需选用自顶向下分析技术时,首先必须判别所给文法是否是LL(1)文法。因而我们对任给文法需计算FIRST、FOLLOW、SELECT集,进而判别文法是否为LL(1)文法。

FIRST集

1、定义:

设G=(VT,VN,S,P)是上下文无关文法 ,FIRST(α)={a|α→aβ,a∈VT,α,β∈V*}   

特别的,若α→ε,则规定ε∈FIRST(α).

2、根据定义求解FIRST集(对每一文法符号X∈VN 计算FIRST(X)):

(1). 若X∈VT,则FIRST(X)={X}。

(2). 若X∈VN,且有产生式X→a…,a∈VT, 则 a∈FIRST(X)X→ε,则ε∈FIRST(X)。 

(3). X→Y…是一个产生式且Y ∈VN  则把FIRST(Y)中的所有非空符号串ε元素都加入到FIRST(X)中。

(4).若X∈VN;Y1,Y2,…,Yi∈VN,且有产生式X→Y1 Y2 … Yn;当Y1 Y2 … Yn-1→ε时,则FIRST(Y1)、FIRST(Y2)、…、FIRST(Yn-1)的所有非空元素和FIRST(Yn) 包含在FIRST(X)中。

(5).当(4)中所有Yi→ε,(i=1,2,…n),则

FIRST(X)=(FIRST(Y1)-{ε})∪(FIRST(Y2)- {ε}∪…∪(FIRST(Yn) -{ε})∪{ε}

反复使用上述(2)~(5)步直到每个符号的FIRST集合不再增大为止。

 

FOLLOW集

1、定义:

设G=(VT,VN,S,P)是上下文无关文法,A∈VN,S是开始符号  

需要注意的是,FOLLOW(A)集是针对非终结符A的,集合中的元素是终结符,是A的全部后跟符号的集合,当A是文法G的开始符(识别符)时,把‘#也加入其中’

2、根据定义求解FOLLOW集(对每一文法符号S∈VN 计算FOLLOW(S)):

(1). 设S为文法中开始符号,把{#}加入FOLLOW(S)中(这里“#”为句子括号)。

(2). 若A→αBβ是一个产生式,则把FIRST(β)的非空元素加入FOLLOW(B)中。如果β→ε则把FOLLOW(A)也加入FOLLOW(B)中。

(3).反复使用(b)直到每个非终结符的FOLLOW集不再增大为止。

 

SELECT集

1、定义:

给定上下文无关文法的产生式A→α, A∈VN,α∈V*:

若α不能推导出ε,则SELECT(A→α)=FIRST(α)   

如果α能推导出ε则:SELECT(A→α)=(FIRST(α) –{ε})∪FOLLOW(A)

需要注意的是,SELECT集是针对产生式而言的。

 

概要设计

首先运行源程序,在界面相应文本区依次输入文法G[S]的产生式,显示到文本区域t1 (JTextArea类型) 中,对产生式无顺序要求,但不允许输入非法的产生式,例如空。输入完成后,进入文法分析阶段。

文法分析程序的功能是对文本区t1的所有字符串从头到尾逐一扫描,然后将读到的字符串(如S→AB)拆分成每个字符进行分析,“→”左侧为产生式的左部(非终结符Vn),右侧为产生式的右部,一般的产生式为二型文法,即左侧只允许输入一个大写字母,右侧输入的字符串中的非大写字母(小写字母a~z,各种符号)为终结符Vt。

LL(1)文法分析的过程如下:

(1) 计算FIRST集。

(2) 计算FOLLOW集。

(3) 计算SELECT集。

(4) 判定是否为LL(1)文法。

 

在求解FIRST集时定义了如下数据结构:

String Vn[] = null;        //所有非终结符集合

int Vnnum;              //非终结符个数

int firstComplete[] = null; // 存储已判断过 first 的数据

char first[][] = null;       // 存储最后 first 结果

first = new char[Vnnum][100];

String s[] = t1.getText().split("\n");  //依次读取文本框中的每条产生式

char yn_null[] = null; // 存储能否推出空

String firstVn[] = null;      // 进行first判断的Vn集合

 

在求解FOLLOW集时定义了如下数据结构:

  • 4
    点赞
  • 35
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
程序的所用的存储结构都是string类型的,最主要的存储文法的数据结构为自定义结构,里面包括一个产生式的左部,右部以及select集合,至于非终结符的first和follow集合,则是定义了一个string类型的数组进行存储。 本程序的求first,follow,select集合的算法即为书上所介绍的方法,即求first的集合时,只看本产生式,求follow集合时,要进行递归查找一个非终结符的所有后跟字符,求select其实就是对first与follow集合的运算,最终根据所有的select集合,便可以判断文法是否为LL(1)文法。 对于不是LL(1)文法的产生式,本程序判断后进行转换,先进行消除左递归,然后提取左公因子,在这两步的每一步结束之后,都要对产生式进行整合,去掉空存储,去掉无法到达的产生式,将select全部置空。 每进行一次非LL(1)到LL(1)的转换之后,都要对其文法性质进行判断,如果是LL(1),则跳出,不是则继续,但是当循环一定次数之后仍不是,程序判定其无法转换,也要跳出。 其中还有对第一个非终结字符的右部替换与否进行选择,原因是,有些通过替换就可以很方便的进行转换,这个要通过人为进行输入。 提取公因子中也有上一段所说的类似的判断机制,目的是为了防止文法的左公因子无法提取完的情况出现。 最终有三种结果,一种是是LL(1)文法,一种是不是LL(1),但是经过转换变成了LL(1),还有一种是经过转换也无法变成LL(1)。 输入文本格式样例: A A->ad A->Bc B->aA B->bB

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值