- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章C++编译原理之求解First集合由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
目的:熟练掌握自上而下的语法分析方法,并能用程序实现.
要求:
例如,使用的文法如下: 编写First函数,实现其求解过程.
E -> TE' E' -> +TE' | # T -> FT' T' -> *FT' | # F -> (E) | id end 。
提示:
不针对特定文法,编写求first函数.
。
A -> a, 则将 a 加入 First(A)中 A -> Y1Y2・・・Yn 。
将 First(Y1) 除空串外的字符加入到First(A)中,若 1 =< i < n - 1,Y1,Y2, Yi中均含有空串,则将First(Yi + 1)加入到First(A)中,若Y1,Y2,・・・,Yn都有空串,则将空串加入到First(A)中 。
First(a) = {a} 。
。
将输入格式化(扫描输入) 将产生式转换为哈希map:
。
/*** @brief Function for generating set of First(a)* @author 立秋小猪* @time: 2021/10/13* @notice: 要求产生体句型不得有空格* 左递归的产生体中必须有空串(必须能够终结)* char '#' act as varepsilon * **/#include <iostream>#include <unordered_map>#include <vector>#include <string>#include <fstream>#include <unordered_set>using namespace std;unordered_map<string, vector<string>> P; //产生式P的集合void scan(){ //scan函数实现从文件扫描文法,将对应的产生式加入到映射P中 fstream fs; string input; fs.open("lan.txt"); if(!fs.is_open()){ // 文件打开失败 cout << "Error: Could not open the file" << endl; exit(-1); } fs >> input; while(input != "end"){ string VN = input; // 产生式的非终结符 fs >> input; //跳过推导符号 if (input != "->" && input != "→"){ cout << "Error: undefined symbol [" << input << "]" << endl; exit(-2); } fs >> input; //产生体拆开后加入到set集合中,默认推导符号后必有一个产生体 P[VN].emplace_back(input); while( fs >> input && input == "|"){ fs >> input; P[VN].emplace_back(input); } }}// void generate(){// }unordered_set<char> First(const string& str){ // 终结符以及空串情况下, whether has the VN or not if(str == "" || str == "#" || P.find(str) == P.end()) return {}; if(!(str[0] >= 'A' && str[0] <= 'Z')) return {str[0]}; vector<string> bodys = P[str]; // str -> bodys unordered_set<char> res = {}; for(auto &s: bodys){ bool hasBlank = true;//是否含有空串,是否继续读产生体 for (int i = 0; i < s.size() && hasBlank; ++i){ if(s[i] >= 'A' && s[i] <= 'Z'){//是否为终结符 unordered_set<char> temp = {};//递归的临时集 string next; if(i < s.size() - 1 && s[i + 1] == '\''){ // 大写字母 + ' 的非终结符 next = s.substr(i, 2); ++i; }else{ //仅仅是大写字母的终结符 next = s[i]; } if(next != str){ //避免无限递归,默认自身是含有空串(hasBlank为True) temp = First(next); //递归求解 hasBlank = false; //先默认temp中没有空串 for(auto &c : temp) if(c == '#') hasBlank = true;//temp中发现了空串 else res.emplace(c); } }else{ res.emplace(s[i]); hasBlank = false;//默认连接的终结符不为空,故此终结符后不会再有新元素加入First集 } } if(hasBlank) //产生体中所有非终结符都包含空串,则将空串加入first集中 res.emplace('#'); } return res;}int main(){ // unordered_map<string, vector<char>> First; //First集合 scan(); cout << "输入的产生式如下:\n" << "********************************\n"; for(auto &[vn, bodys]: P){ cout << vn << " -> " << bodys[0]; for (int i = 1; i < bodys.size(); ++i) cout << " | " << bodys[i]; cout << endl; } cout << "********************************\n"; for(auto &[vn,_]: P){ unordered_set<char> f = First(vn); cout << "First(" << vn << ") : "; auto iter = f.begin(); if(iter != f.end()){ cout << *iter; while(++iter != f.end()){ cout << " , " << *iter; } } cout << endl; } return 0;}
E -> TE'E' -> +TE' | #T -> FT'T' -> *FT' | #F -> (E) | idend
运行结果 。
S -> SaRb | #R -> RSQ | #Q -> eend
运行结果 。
到此这篇关于C++/编译原理之求解First集合的文章就介绍到这了,更多相关C++ 求解First集合内容请搜索我以前的文章或继续浏览下面的相关文章希望大家以后多多支持我! 。
原文链接:https://www.cnblogs.com/liqiuxiaozhu/p/15403976.html 。
最后此篇关于C++编译原理之求解First集合的文章就讲到这里了,如果你想了解更多关于C++编译原理之求解First集合的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
目前,我正在为网络开发类(class)做作业。 这些是说明:第一行和首字下沉样式Jakob 希望文章的第一行以小写大写字母显示。转到 First Line and Drop Cap Styles 部分
.first() 方法是在 jQuery 1.4 中添加的。 :first 选择器自 1.0 以来就已存在。 来自文档: :first The :first pseudo-class is equiv
我正在审查现有的 ASP.NET MVC (5.2.3) EF (6.1.3) 项目。 该项目使用 ASP.NET Identity,我检查了 web.config 中的 2 个连接字符串,一个用于
为什么人们使用 mid=first+(last-first)/2 而不是 (first+last)/2,在二进制搜索的情况下)两者有区别吗。如果有,请告诉我,因为我无法理解其中的区别。 最佳答案 如果
为什么人们使用 mid=first+(last-first)/2 而不是 (first+last)/2,在二进制搜索的情况下)两者有区别吗。如果有,请告诉我,因为我无法理解其中的区别。 最佳答案 如果
for(auto it = M.begin(); it!=M.end();it++) { coutfirstsecondsecond == 1) return it->firs
我试图从第二个循环中获取循环的第一项。 我知道我得到了这样的@key @../key 但@first 似乎不像@../first 那样工作 有什么想法吗? 问候 最佳答案 首先,无论是否在嵌套 blo
var tab1 = $('.tabs a:first-child').attr('href'); alert(tab1); .. 尽管同一页面上有两个 div.switch,但仅匹配一个。第二个位于
我想知道如何将节点*变量 NODE 分配给结构内的数据? struct node { int info; struct node *link; }; typedef struct nod
我有两个段落包含在一个 div 中。我想让第一段的文字变大一点,但使用 :first-child 并不能像我所说的那样工作。看不出有什么问题。
我有一个 ul li 列表 Parent child1 child2
我有三个表,即员工、部门和申诉。 Employees 表有超过一百万条记录。我需要找到员工的详细信息、他/她的部门以及他/她提出的申诉。 我可以想到以下两个查询来查找结果: 1。先过滤记录,只获取需要
我有三个表,即员工、部门和申诉。 Employees 表有超过一百万条记录。我需要找到员工的详细信息、他/她的部门以及他/她提出的申诉。 我可以想到以下两个查询来查找结果: 1。先过滤记录,只获取需要
这有什么区别吗: myList.Where(item => item == 0).First(); 还有这个: myList.First(item => item == 0); 后者对我来说更有意义,
我分不清 element:first-child 之间的区别和 element:first-of-type 例如,你有一个 div div:first-child → 全部 元素是其父元素的第一个子元
当我遇到一个奇怪的情况时,我正在研究 CSS 选择器。 如果我使用 :first-child 伪元素,我需要在它前面加上一个空格才能工作,否则它将无法工作。然而 :first-letter 伪元素的情
请考虑以下字符串数组: let strings = ["str1", "str2", "str10", "str20"] 假设需要获取包含 5 个字符的第一个元素 (String),我可以使用 fil
让我们假设我们要开始新项目 - 包含一些业务逻辑的应用程序、ASP.NET 上的用户界面、WPF 或两者。我们想使用 ORM 或 DAL 代码生成器并在 .NET 类中实现我们的业务逻辑。我们可以通过
我有一种树系统。我想做的是给所有 parent 一个 margin ,除了第一个。这是我的 HTML: Test
我分不清 element:first-child 之间的区别和 element:first-of-type 例如,你有一个 div div:first-child → 全部 元素是其父元素的第一个子元
我是一名优秀的程序员,十分优秀!