上篇文章【编译原理】语法分析——自上向下分析 分析了LL1语法,文章最后说给出栗子,现在补上去。

说明:

  • 这个语法分析器是利用LL1分析方法实现的。
  • 预测分析表和终结符以及非终结符都是针对一个特定文法定义好的。
  • 输入的分析串必须以 # 开头和结尾。

原始文法:

E -> E + T | T
T -> T * T | F
F -> (E) | i

消除左递归之后

E -> TE'
E' -> +TE' | e
T -> FT'
T' -> *FT' | e
F -> (E) | i //备注
e 表示 一步赛楞
E' 在实际程序中用Z代替
T' 在实际程序中用Y代替

程序如下:

#include<iostream>
#include<vector>
#include<string>
#include<stack> #define BEGIN_SYB 'E' class Tab { //预测分析表中单个产生式
public:
Tab(char n,char e,std::string p):noend(n),end(e),prod(p) {}; char noend;
char end;
std::string prod;
}; std::vector<Tab> pTab; //预测分析表
std::stack<char> pStack; //栈
std::string pStr; //输入串
int index = 0; //输入串的下标
int num = 0; //下标
char X; //从栈顶取出的符号
char a; //字符串目前分析到的位置
std::vector<char> VT{'i','+','*','(',')','#'}; //终结符号集合
std::vector<char> VN{'E','Z','T','Y','F'}; //非终结符号集合 int
isPartVT(char ch)
{
for(auto u:VT) {
if(u == ch) {
return 1;
}
}
return 0;
} int
isPartVN(char ch)
{
for(auto u:VN) {
if(u == ch) {
return 1;
}
}
return 0;
} void
initpTab()
{
pTab.push_back(Tab('E','i',"TZ")); //Z表示E'
pTab.push_back(Tab('E','(',"TZ"));
pTab.push_back(Tab('Z','+',"+TZ"));
pTab.push_back(Tab('Z',')',"$"));
pTab.push_back(Tab('Z','#',"$"));
pTab.push_back(Tab('T','i',"FY"));
pTab.push_back(Tab('T','(',"FY")); //Y表示T'
pTab.push_back(Tab('Y','+',"$"));
pTab.push_back(Tab('Y','*',"*FY"));
pTab.push_back(Tab('Y',')',"$"));
pTab.push_back(Tab('Y','#',"$"));
pTab.push_back(Tab('F','i',"i"));
pTab.push_back(Tab('F','(',"(E)"));
} void
printStack()
{
std::stack<char> pTemp(pStack);
while(pTemp.size() != 0) {
std::cout << pTemp.top() << " ";
pTemp.pop();
}
std::cout << " ";
} int
synaly()
{
pStack.push(pStr[index++]);
pStack.push(BEGIN_SYB);
a = pStr[index++];
bool flag = true;
while(flag) {
std::cout << num++ << " "; //输出行号
printStack();
if(pStack.size() != 0) {
X = pStack.top(); //将符号栈顶给X
pStack.pop();
} if(isPartVT(X)) { //如果是终结符
if(X == '#' && a == '#') {
flag = false;
}else if(X == a){
std::cout << std::endl;
a = pStr[index++];
}else {
std::cout << "EROOR!" << std::endl;
return 0;
} }else if (X == '#') {
if(X == a) {
flag = false;
}else {
std::cout << "EROOR!" << std::endl;
return 0;
} }else if (isPartVN(X) && isPartVT(a)) { //如果非终结符
int type = 0;
for(auto u:pTab) {
if(u.noend == X && u.end == a) {
std::string tempProd = u.prod;
std::cout << X << "->" << tempProd << std::endl;
if(tempProd == "$") {
type = 1;
break;
}else {
for(int i = tempProd.size()-1;i >= 0;--i) {
pStack.push(tempProd[i]);
}
type = 1;
break;
}
}
}
if(type != 1) {
std::cout << "EROOR!" << std::endl;
return 0; } }else {
std::cout << "EROOR!" << std::endl;
return 0;
} }
return 1;
} int main(int argc,char *argv[])
{
initpTab();
std::cout << "请输入语法串:";
std::cin >> pStr;
std::cout << "步骤" << " " << "符号栈" << " " << "所用产生式" << std::endl;
int flag = synaly();
std::cout << std::endl;
if(flag == 1) {
std::cout << "分析成功~~" << std::endl;
}else {
std::cout << "分析失败~~" << std::endl;
} return 0;
}

【编译原理】LL1文法语法分析器的更多相关文章

  1. 编译原理LL1文法分析树(绘图过程)算法实现

    import hjzgg.analysistable.AnalysisTable; import hjzgg.first.First; import hjzgg.follow.Follow; impo ...

  2. 编译原理LL1文法分析表算法实现

    import hjzgg.first.First; import hjzgg.follow.Follow; import hjzgg.tablenode.TableNode; import hjzgg ...

  3. 编译原理LL1文法Follow集算法实现

    import hjzgg.first.First; import java.util.LinkedHashMap; import java.util.Map; import java.util.Set ...

  4. 编译原理 LL1文法First集算法实现

    import java.util.LinkedHashMap; import java.util.Map; import java.util.Set; import java.util.TreeMap ...

  5. Compiler Theory(编译原理)、词法/语法/AST/中间代码优化在Webshell检测上的应用

    catalog . 引论 . 构建一个编译器的相关科学 . 程序设计语言基础 . 一个简单的语法制导翻译器 . 简单表达式的翻译器(源代码示例) . 词法分析 . 生成中间代码 . 词法分析器的实现 ...

  6. 编译原理简单语法分析器(first,follow,分析表)源码下载

    编译原理(简单语法分析器下载) http://files.cnblogs.com/files/hujunzheng/%E5%8A%A0%E5%85%A5%E5%90%8C%E6%AD%A5%E7%AC ...

  7. <编译原理 - 函数绘图语言解释器(2)语法分析器 - python>

    <编译原理 - 函数绘图语言解释器(2)语法分析器 - python> 背景 编译原理上机实现一个对函数绘图语言的解释器 - 用除C外的不同种语言实现 设计思路: 设计函数绘图语言的文法, ...

  8. 【编译原理】c++实现自下而上语法分析器

    写在前面:本博客为本人原创,严禁任何形式的转载!本博客只允许放在博客园(.cnblogs.com),如果您在其他网站看到这篇博文,请通过下面这个唯一的合法链接转到原文! 本博客全网唯一合法URL:ht ...

  9. 【编译原理】c++实现自上而下语法分析器

    写在前面:本博客为本人原创,严禁任何形式的转载!本博客只允许放在博客园(.cnblogs.com),如果您在其他网站看到这篇博文,请通过下面这个唯一的合法链接转到原文! 本博客全网唯一合法URL:ht ...

随机推荐

  1. (转载)深入理解MDL元数据锁

    作者:MySQL技术本文为作者原创,转载请注明出处:https://www.cnblogs.com/kunjian/p/11993708.html 前言: 当你在MySQL中执行一条SQL时,语句并没 ...

  2. Windows 10 64位操作系统 下安装、启动测试python pycharm

    一.下载python3.7.7安装包 1:详细下载安装版本可见官网:https://www.python.org/downloads/release/python-373/ 2:百度盘分享连接:htt ...

  3. css 圆形脉冲动画

    需求: 项目需要在3D场景增加动画按钮,直接添加到场景时 当场景过大的时候 .加载比较麻烦 因在找资料时发现这玩意居然要付费.故做此记录, 效果: 参考: 1.https://www.jiangwei ...

  4. three.js 添加html内容、文本

    需求: 1.在场景内添加html元素并动态更新 2.html内容需跟随场景变化 方案: 新加方案:https://www.zhihu.com/question/49929467/answer/1186 ...

  5. P5956-[POI2017]Podzielno【数学】

    正题 题目链接:https://www.luogu.com.cn/problem/P5956 题目大意 \(B\)进制下,给出序列\(a\),\(a_i\)表示数字\(i\)有多少个.求一个最大的\( ...

  6. P4457-[BJOI2018]治疗之雨【期望dp,高斯消元】

    正题 题目链接:https://www.luogu.com.cn/problem/P4457 题目大意 开始一个人最大生命值为\(n\),剩余\(hp\)点生命,然后每个时刻如果生命值没有满那么有\( ...

  7. Maven----将手动下载的jar包以命令行的方式安装到本地MavenRepository中

    1.情景再现:准备实现SprintBoot的热部署功能,因没有对应jar时,在Eclipse中mvn install 会报错: 报错信息: [INFO] --- spring-boot-maven-p ...

  8. 【深度学习】线性回归(Linear Regression)——原理、均方损失、小批量随机梯度下降

    1. 线性回归 回归(regression)问题指一类为一个或多个自变量与因变量之间关系建模的方法,通常用来表示输入和输出之间的关系. 机器学习领域中多数问题都与预测相关,当我们想预测一个数值时,就会 ...

  9. SpringBoot碰到的疑问或问题

    1.@ResponseBody 和 @RequestBody 的区别 @ResponseBody是作用在方法上的,@ResponseBody 表示该方法的返回结果直接写入 HTTP response ...

  10. 七牛云的 python sdk 是如何 批量删除资源的

    今天做项目的时候用到七牛云,关于对资源的操作是在后端做的,用的SDK,但是,在网上没找到详细的解析,官方文档也没有太详细的解说,所以无奈只好看下源码 这里做一下简单的记录 from qiniu imp ...