编译原理-求FOLLOW集合

2024-03-01 16:50
文章标签 编译 原理 集合 follow

本文主要是介绍编译原理-求FOLLOW集合,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

  • 为什么需要求FIRST集合:因为一个产生式存在多个候选式,选择哪一个候选式是不确定的,所以这就产生了回溯。回溯需要消耗大量的计算、存储空间,所以我们需要消除回溯。而消除回溯的其中一种方法叫作“预测”,即根据栈顶非终结符去预测后面的候选式,那预测方法就是求第一个非终结符,来判断是否和读头匹配,以达到预测的效果
  • 为什么需要求FOLLOW集合:求FOLLOW集合的目的和FIRST集合的目的是一样的,但是应该对的情形不一样,当出现了下列情形时,求FIRST集合已经达不到要求了

微信公众号:JavaWeb架构师

但是我们观察后发现,当F推出ε时,下面的E推出的a是能进行匹配的,在F这个位置求到a,就是在求FOLLOW集合

FOLLOW集合定义

  • 文字定义:FOLLOW(A)集合是所有紧跟A之后的终结符或#所组成的集合(#是句尾的标志),称FOLLOW(A)是A的随符集

  • 公式定义:
    设G是上下文无关文法,A ∈ VN,S是开始符
    微信公众号:JavaWeb架构师
    则:
    微信公众号:JavaWeb架构师

  • 注意

    • 当A是最右部的时候,# 加入 FOLLOE(A)

计算FOLLOW集合步骤

  • 求解非终结符A的随符集FOLLOW(A)
  • 1)对S,将 # 加入 FOLLOW(S),然后再按后面的处理
  • 2)若B → αAβ是G的产生式,则将FIRST(β) - ε 加入FOLLOW(A)
  • 3)若B → αA是G的产生式,或B → αAβ是G的产生式(β 多次推导后得到ε ),则将FOLLOW(B) 加入到FOLLOW(A) 【因为把B用αA替换之后,B后面紧跟的字符就是A后面紧跟的字符】
  • 4)反复使用2)-3),直到FOLLOW集合不再增大为止

  • 注意

    • 这里的文法G必须是消除左递归且提取了左因子(点击查看提取左因子的办法)

例题

  • 对如下文法G,E为开始符,求E、T、F的随符集
1.E  →  TE'
2.E'  →  +TE'
3.E'  →  ε
4.T  →  FT'
5.T'  →  *FT'
6.T'  →  ε
7.F  →  i
8.F  →  (E)

解:

---------------FOLLOW(E)---------------
FOLLOW(E) = {)} ;   // 8. 规则2)
又 ∵  E是开始符,所以FOLLOW(E) = { #,) } ;     // 规则1)---------------FOLLOW(T)---------------
FOLLOW(T) = FIRST(E') - ε; // 1.  规则2)
FIRST(E') = {+,ε};     // 2. 3.  
当E' = ε时,E  →  TE' 就是 E  →  T,所以FOLLOW(E)也要加入FOLLOW(T),
∴ FOLLOW(T) = {+, #,)}
同理,FOLLOW(E')也需要加入到FOLLOW(T)中,
FOLLOW(E') = FOLLOW(E) = { #, )};   // 1. 规则3)
∴ FOLLOW(T) = {+, #,)}---------------FOLLOW(F)---------------
与FOLLOW(F)相关的式子有:
T'  →  *FT'
T  →  FT'先,T'  →  *FT'
∴ FOLLOW(F) = FIRST(T') - ε;
FIRST(T') = {*,ε}
当T'为ε时,也要把FOLLOW(T')加入FOLLOW(F)中,
FOLLOW(T') = {+, #,)},∴ FOLLOW(F) =  {+,*, #,)}再,T  →  FT',
∴ FOLLOW(F) = FIRST(T') - ε;
当T'为ε时,也要把FOLLOW(T)加入FOLLOW(F)中,
∴ FOLLOW(F) =  {+,*, #,)}

    • 形如 T’ → *FT’,求FOLLOW(T’)时,针对本句,可以忽略3),因为是还是再求FOLLOW(T’)
    • 对于产生了ε的,一定要记得3)
    • FIRST求解的时候,关注点在左部,进行推导;FOLLOW求解的时候,关注点在右部,看跟随

其它

  • 课件下载:
关注下方微信公众号,
回复:
FOLLOW.code
  • 欢迎加入交流群:451826376

  • 更多信息:www.itcourse.top

完整教程PDF版本下载

这篇关于编译原理-求FOLLOW集合的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/763141

相关文章

SpringBoot分段处理List集合多线程批量插入数据方式

《SpringBoot分段处理List集合多线程批量插入数据方式》文章介绍如何处理大数据量List批量插入数据库的优化方案:通过拆分List并分配独立线程处理,结合Spring线程池与异步方法提升效率... 目录项目场景解决方案1.实体类2.Mapper3.spring容器注入线程池bejsan对象4.创建

ShardingProxy读写分离之原理、配置与实践过程

《ShardingProxy读写分离之原理、配置与实践过程》ShardingProxy是ApacheShardingSphere的数据库中间件,通过三层架构实现读写分离,解决高并发场景下数据库性能瓶... 目录一、ShardingProxy技术定位与读写分离核心价值1.1 技术定位1.2 读写分离核心价值二

深度解析Python中递归下降解析器的原理与实现

《深度解析Python中递归下降解析器的原理与实现》在编译器设计、配置文件处理和数据转换领域,递归下降解析器是最常用且最直观的解析技术,本文将详细介绍递归下降解析器的原理与实现,感兴趣的小伙伴可以跟随... 目录引言:解析器的核心价值一、递归下降解析器基础1.1 核心概念解析1.2 基本架构二、简单算术表达

深入浅出Spring中的@Autowired自动注入的工作原理及实践应用

《深入浅出Spring中的@Autowired自动注入的工作原理及实践应用》在Spring框架的学习旅程中,@Autowired无疑是一个高频出现却又让初学者头疼的注解,它看似简单,却蕴含着Sprin... 目录深入浅出Spring中的@Autowired:自动注入的奥秘什么是依赖注入?@Autowired

从原理到实战解析Java Stream 的并行流性能优化

《从原理到实战解析JavaStream的并行流性能优化》本文给大家介绍JavaStream的并行流性能优化:从原理到实战的全攻略,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的... 目录一、并行流的核心原理与适用场景二、性能优化的核心策略1. 合理设置并行度:打破默认阈值2. 避免装箱

Python中的filter() 函数的工作原理及应用技巧

《Python中的filter()函数的工作原理及应用技巧》Python的filter()函数用于筛选序列元素,返回迭代器,适合函数式编程,相比列表推导式,内存更优,尤其适用于大数据集,结合lamb... 目录前言一、基本概念基本语法二、使用方式1. 使用 lambda 函数2. 使用普通函数3. 使用 N

MyBatis-Plus 与 Spring Boot 集成原理实战示例

《MyBatis-Plus与SpringBoot集成原理实战示例》MyBatis-Plus通过自动配置与核心组件集成SpringBoot实现零配置,提供分页、逻辑删除等插件化功能,增强MyBa... 目录 一、MyBATis-Plus 简介 二、集成方式(Spring Boot)1. 引入依赖 三、核心机制

Java集合中的链表与结构详解

《Java集合中的链表与结构详解》链表是一种物理存储结构上非连续的存储结构,数据元素的逻辑顺序的通过链表中的引用链接次序实现,文章对比ArrayList与LinkedList的结构差异,详细讲解了链表... 目录一、链表概念与结构二、当向单链表的实现2.1 准备工作2.2 初始化链表2.3 打印数据、链表长

redis和redission分布式锁原理及区别说明

《redis和redission分布式锁原理及区别说明》文章对比了synchronized、乐观锁、Redis分布式锁及Redission锁的原理与区别,指出在集群环境下synchronized失效,... 目录Redis和redission分布式锁原理及区别1、有的同伴想到了synchronized关键字

Linux中的HTTPS协议原理分析

《Linux中的HTTPS协议原理分析》文章解释了HTTPS的必要性:HTTP明文传输易被篡改和劫持,HTTPS通过非对称加密协商对称密钥、CA证书认证和混合加密机制,有效防范中间人攻击,保障通信安全... 目录一、什么是加密和解密?二、为什么需要加密?三、常见的加密方式3.1 对称加密3.2非对称加密四、