编译原理 LL(1)文法判别方法

实验目的与内容

理解LL(1)文法的判定方法,进一步掌握文法的改造原理(左递归消去、左公共因子提取),FIRST集、FOLLOW集的构造方法,预测分析表的的构造算法。
针对任意的文法,编写相应的左递归消除、左公共因子提取程序,求解相应的FIRST、FOLLOW集,构造预测分析表。
判断LL(1)文法部分:

  1. 输入:文法
  2. 处理:左递归消除、左公共因子提取,FIRST、FOLLOW等集合构造,判断LL(1)
  3. 输出:是LL(1)的情况输出预测分析表,否则判断不是LL(1)

程序总体设计思路和框架

使用公共map储存FIRST,FOLLOW,SELECT集,并记录开始符号、终结符号和非终结符号。
主要函数:
public static String Init(File file)
用于初始化分析,根据所给的路径读取txt中的文法,文法左式和右式用空格隔开(输入“E TX”代表“E->TX”)。确定终结符号和非终结符号以及开始符号。
public static boolean dofirst()
该函数用于找到文法非终结符号的FIRST集,根据定义求解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集合不再增大为止。
如果存在左递归则输出false,以下函数都不再执行
public static void dofollow()
该函数用于找到文法的FOLLOW集,根据定义求解FOLLOW集(对每一文法符号S∈VN 计算FOLLOW(S)):
(1). 设S为文法中开始符号,把{$}加入FOLLOW(S)中。
(2). 若A→αBβ是一个产生式,则把FIRST(β)的非空元素加入FOLLOW(B)中。如果β→ε则把FOLLOW(A)也加入FOLLOW(B)中。
(3).反复使用(b)直到每个非终结符的FOLLOW集不再增大为止。
public static void doSelect()
该函数用于找到文法的SELECT集,
若α不能推导出ε,则SELECT(A→α)=FIRST(α)   
如果α能推导出ε则:SELECT(A→α)=(FIRST(α) –{ε})∪FOLLOW(A)
public static boolean checkLL1()
该函数用于检查该文法是否为LL(1)文法
这里使用一个finalselect储存同一个非终结符号的SELECT集,并且将其转换为不能有重复元素的set集合,如果长度变小则不是LL(1)文法,如果长度相同则是LL(1)文法。

主要数据结构

public static String str="";
public static Map<Character,List<Character>> first = new HashMap<>();
public static Map<Character,List<Character>> follow = new HashMap<>();
public static Map<String,List<Character>> select = new HashMap<>();
public static Map<Character,List<Character>> finalselect = new HashMap<>();//储存同一个非终结符号的select集
public static List<Character> unfinal = new LinkedList<>();//非终结符号
public static List<Character> ifinal = new LinkedList<>();//终结符号
public static char begin;//开始符号
public static Map<Character,List<String>> relation = new HashMap<>();//储存文法

代码

import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
import java.util.*;


public class IsLL1 {
   //LL(1)文法判别方法

    public static String str="";
    public static Map<Character,List<Character>> first = new HashMap<>();
    public static Map<Character,List<Character>> follow = new HashMap<>();
    public static Map<String,List<Character>> select = new HashMap<>();
    public static Map<Character,List<Character>> finalselect = new HashMap<>();//储存同一个非终结符号的select集
    public static List<Character> unfinal = new LinkedList<>();//非终结符号
    public static List<Character> ifinal = new LinkedList<>();//终结符号
    public static char begin;//开始符号
    public static Map<Character,List<String>> relation = new HashMap<>();//储存文法
    public static String Init(File file){
   

        try{
   
            BufferedReader br = new BufferedReader(new FileReader(file));//构造一个BufferedReader类来读取文件
            String s = null;
            int is = 1;
            while((s = br.readLine())!=null){
   //使用readLine方法,一次读一行
                String[] s1 = s.split(" ");//按空格分隔,要求输入“E TX”代表“E->TX”
                if(!unfinal.contains(s1[0].charAt(0)))
                    unfinal.add(s1[0].charAt(0));//加入左侧非终结符号
                char[] chars = s1[1]
  • 1
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 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、付费专栏及课程。

余额充值