简单的FIRST+集演示程序
/*
* 该程序用于计算某个非终结符的 FIRST+ 集合
* RexfieldVon
* 2013年6月30日16:02:47
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h> /* 三级指针
* 第一级指向整个产生式组
* 第二级指向单个产生式
* 第三级指向产生式符号单元
* 约定:所有的大写字母为非终结符
* 假设:无左递归、FIRST集中不会出现重复符号
*/
char*** GrammerRule; /*
* 初始化文法序列
*/
void InitizationGrammerRule()
{
// 分配表头空间
GrammerRule = (char***)malloc(sizeof(int) * );
memset(GrammerRule, '\0', sizeof(int) * );
// 分配文法空间并写入产生式
// G -> E
GrammerRule['G'] = (char**)malloc(sizeof(int) * );
GrammerRule['G'][] = (char*)malloc();
memcpy(GrammerRule['G'][], "E\0", ); // E
GrammerRule['G'][] = NULL;
// E -> T F
GrammerRule['E'] = (char**)malloc(sizeof(int) * );
GrammerRule['E'][] = (char*)malloc();
memcpy(GrammerRule['E'][], "TF\0", ); // T F
GrammerRule['E'][] = NULL;
// F -> '+' T F | '-' T F | e
GrammerRule['F'] = (char**)malloc(sizeof(int) * );
GrammerRule['F'][] = (char*)malloc();
memcpy(GrammerRule['F'][], "+TF\0", ); // '+' T F
GrammerRule['F'][] = (char*)malloc();
memcpy(GrammerRule['F'][], "-TF\0", ); // '-' T F
GrammerRule['F'][] = (char*)malloc();
memcpy(GrammerRule['F'][], "\0", ); // e (该产生式存在但是为空)
GrammerRule['F'][] = NULL;
// T -> A U
GrammerRule['T'] = (char**)malloc(sizeof(int) * );
GrammerRule['T'][] = (char*)malloc();
memcpy(GrammerRule['T'][], "AU\0", ); // A U
GrammerRule['T'][] = NULL;
// U -> '*' A U | '/' A U | e
GrammerRule['U'] = (char**)malloc(sizeof(int) * );
GrammerRule['U'][] = (char*)malloc();
memcpy(GrammerRule['U'][], "*AU\0", ); // '*' A U
GrammerRule['U'][] = (char*)malloc();
memcpy(GrammerRule['U'][], "/AU\0", ); // '/' A U
GrammerRule['U'][] = (char*)malloc();
memcpy(GrammerRule['U'][], "\0", ); // e (该产生式存在但是为空)
GrammerRule['U'][] = NULL;
// A -> '(' E ')' | d | n
GrammerRule['A'] = (char**)malloc(sizeof(int) * );
GrammerRule['A'][] = (char*)malloc();
memcpy(GrammerRule['A'][], "(E)\0", ); // '(' E ')'
GrammerRule['A'][] = (char*)malloc();
memcpy(GrammerRule['A'][], "d\0", ); // d
GrammerRule['A'][] = (char*)malloc();
memcpy(GrammerRule['A'][], "n\0", ); // n
GrammerRule['A'][] = NULL;
} /*
* 取得终结符数量
*/
int GetTerminalCount()
{
int i, TerminalCount = ;
for (i = ; i < ; i++)
{
if (GrammerRule[i] != NULL)
{
int k = ;
while (GrammerRule[i][k] != NULL)
{
int n = ;
while (GrammerRule[i][k][n] != '\0')
{
char c = GrammerRule[i][k][n];
if (c < 'A' || c > 'Z')
{
TerminalCount++;
}
n++;
}
k++;
}
}
}
return TerminalCount;
} /*
* 递归取得 FIRST 集
* Token : char 需要打印的符号
* FIRST : char* FIRST集
* Ptr : int* FIRST集的位置指针
*/
void GetFIRST(char Token, char *FIRST, int *Ptr)
{
if (Token >= 'A' && Token <= 'Z' && GrammerRule[Token] != NULL)
{
int i = ;
while (GrammerRule[Token][i] != NULL)
{
GetFIRST(GrammerRule[Token][i++][], FIRST, Ptr);
}
}
else if (Token < 'A' || Token > 'Z')
{
FIRST[*Ptr] = Token;
*Ptr = *Ptr + ;
}
} /*
* 添加符号到 FOLLOW 集
* FOLLOW : char* FOLLOW集
* Ptr : int* FOLLOW集的位置指针
* NewItem : char 将加入的符号
*/
void AddFOLLOWItem(char *FOLLOW, int *Ptr, char NewItem)
{
int i = ;
for (; i < *Ptr; i++)
{
if (FOLLOW[i] == NewItem)
{
return ;
}
}
FOLLOW[*Ptr] = NewItem;
*Ptr = *Ptr + ;
} /*
* 取得 FOLLOW 集
* Unterminal : char 需要打印的非终结符
* FOLLOW : char* FOLLOW集
* Ptr : int* FOLLOW集的位置指针
* TerminalCount : int 终结符数量
*/
void GetFOLLOW(char Unterminal, char *FOLLOW, int *Ptr, int TerminalCount)
{
int RuleIndex, ExprIndex, TokenIndex;
// 开始遍历整个文法
for (RuleIndex = ; RuleIndex < ; RuleIndex++)
{
if (GrammerRule[RuleIndex] == NULL)
{
continue;
}
// 搜索整个文法找到指定的非终结符
for (ExprIndex = ; GrammerRule[RuleIndex][ExprIndex] != ; ExprIndex++)
{
for (TokenIndex = ; GrammerRule[RuleIndex][ExprIndex][TokenIndex] != '\0'; TokenIndex++)
{
if (GrammerRule[RuleIndex][ExprIndex][TokenIndex] == Unterminal)
{
char nc = GrammerRule[RuleIndex][ExprIndex][TokenIndex + ];
if (nc == '\0' && RuleIndex != Unterminal) // 情形三:反复计算:将FOLLOW(P)加入FOLLOW(U)
{
GetFOLLOW((char)RuleIndex, FOLLOW, Ptr, TerminalCount);
}
else if (nc >= 'A' && nc <= 'Z') // 情形二:间接计算:将FIRST(P)加入FOLLOW(U)
{
char *FIRST = (char*)malloc(TerminalCount + );
memset(FIRST, '\0', TerminalCount + );
int FIRSTPtr = , InsertPtr;
GetFIRST(nc, FIRST, &FIRSTPtr);
for (InsertPtr = ; InsertPtr < FIRSTPtr; InsertPtr++)
{
if (FIRST[InsertPtr] != '\0')
{
AddFOLLOWItem(FOLLOW, Ptr, FIRST[InsertPtr]);
}
else // 对于 P->... U B,FOLLOW ← FIRST(B) - <e> + FOLLOW(P)
{
GetFOLLOW((char)RuleIndex, FOLLOW, Ptr, TerminalCount);
}
}
}
else if (nc != '\0') // 情形一:直接计算:将终结符加入FOLLOW(U)
{
AddFOLLOWItem(FOLLOW, Ptr, nc);
}
}
}
}
}
} void GetFIRSTPlus(char Unterminal, int Index, char *FIRSTPlus, int *Ptr, int TerminalCount)
{
if (GrammerRule[Unterminal][Index] != NULL)
{
GetFIRST(GrammerRule[Unterminal][Index][], FIRSTPlus, Ptr);
int i = ;
while (i < *Ptr)
{
if (FIRSTPlus[i++] == '\0')
{
FIRSTPlus[*Ptr] = '\377';
*Ptr = *Ptr + ;
GetFOLLOW(Unterminal, FIRSTPlus, Ptr, TerminalCount);
break;
}
}
}
} /*
* 打印指定非终结符的 FIRST+ 集
* Unterminal : char 需要打印的非终结符
* TerminalCount : int 终结符数量
*/
void PrintUnterminalFIRSTPlus(char Unterminal, int TerminalCount)
{
char *FIRSTPlus = (char*)malloc(TerminalCount + );
memset(FIRSTPlus, '\0', TerminalCount + );
int Ptr, Index = , i;
for (; GrammerRule[Unterminal][Index] != NULL; Index++)
{
Ptr = ;
GetFIRSTPlus(Unterminal, Index, FIRSTPlus, &Ptr, TerminalCount);
printf("FIRST+(%c, %c): ", Unterminal, GrammerRule[Unterminal][Index][]);
for (i = ; i < Ptr; i++)
{
if (FIRSTPlus[i] == '\377')
{
printf("<eof> ");
}
else if (FIRSTPlus[i] == '\0')
{
printf("<e> ");
}
else
{
printf("%c ", FIRSTPlus[i]);
}
}
printf("\n");
}
} int main(int argc, char **argv)
{
InitizationGrammerRule(); // 初始化文法
int TerminalCount = GetTerminalCount();
PrintUnterminalFIRSTPlus('E', TerminalCount);
PrintUnterminalFIRSTPlus('F', TerminalCount);
PrintUnterminalFIRSTPlus('T', TerminalCount);
PrintUnterminalFIRSTPlus('U', TerminalCount);
PrintUnterminalFIRSTPlus('A', TerminalCount);
return ;
}
简单的FIRST+集演示程序的更多相关文章
- 简单的FOLLOW集演示程序
/* * 该程序用于计算某个非终结符的 FOLLOW 集合 * RexfieldVon * 2013年6月30日16:02:47 */ #include <stdio.h> #includ ...
- 简单的FIRST集演示程序
/* * 该程序用于计算某个非终结符的 FIRST 集合 * RexfieldVon * 2013年6月29日19:53:45 * 2013年7月3日22:01:57 修改 GetFIRST 例程以简 ...
- POJ 2524 (简单并查集) Ubiquitous Religions
题意:有编号为1到n的学生,然后有m组调查,每组调查中有a和b,表示该两个学生有同样的宗教信仰,问最多有多少种不同的宗教信仰 简单并查集 //#define LOCAL #include <io ...
- poj1611 简单并查集
The Suspects Time Limit: 1000MS Memory Limit: 20000K Total Submissions: 32781 Accepted: 15902 De ...
- 1213 How Many Tables(简单并查集)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1213 简单并查集,统计单独成树的数量. 代码: #include <stdio.h> #i ...
- 【简单并查集】Farm Irrigation
Farm Irrigation Time Limit : 2000/1000ms (Java/Other) Memory Limit : 65536/32768K (Java/Other) Tot ...
- RocketMQ 简单梳理 及 集群部署笔记【转】
一.RocketMQ 基础知识介绍Apache RocketMQ是阿里开源的一款高性能.高吞吐量.队列模型的消息中间件的分布式消息中间件. 上图是一个典型的消息中间件收发消息的模型,RocketMQ也 ...
- 搭建简单的hadoop集群(译文)
本文翻译翻译自http://hadoop.apache.org/docs/r2.8.0/hadoop-project-dist/hadoop-common/ClusterSetup.html 具体的实 ...
- ACM_“打老虎”的背后(简单并查集)
“打老虎”的背后 Time Limit: 2000/1000ms (Java/Others) Problem Description: “习大大”自担任国家主席以来大力反腐倡廉,各地打击贪腐力度也逐步 ...
随机推荐
- Socket 学习
Socket一般应用模式(服务器端和客户端) 服务器端Socket(至少有两个) ->一个负责接收客户端连接请求(但不负责和客户端通信) ->没成功接收到一个客户端的连接便在服务端生成一个 ...
- WINDOWS批处理命令使用大全
来源:http://www.942dn.com就是爱电脑网 WINDOWS批处理命令使用大全 批处理,也称为批处理脚本,英文译为BATCH,批处理文件后缀BAT就取的前三个字母.它的构成没有固定格式, ...
- JavaScript HTML DOM - 改变CSS
JavaScript HTML DOM - 改变CSS HTML DOM 允许 JavaScript 改变 HTML 元素的样式. 改变 HTML 样式 如需改变 HTML 元素的样式,请使用这个语法 ...
- 浅谈KMP算法及其next[]数组
KMP算法是众多优秀的模式串匹配算法中较早诞生的一个,也是相对最为人所知的一个. 算法实现简单,运行效率高,时间复杂度为O(n+m)(n和m分别为目标串和模式串的长度) 当字符串长度和字符集大小的比值 ...
- Vijos1352 NOI2006 最大获利 最小权闭合图
Orz胡伯涛<最小割模型在信息学竞赛中的应用> 建图方法: 设立源点S和汇点T,S和用户(共M个)连边,载流量为满足其要求的获利 T和中转站(共N个)连边,载流量为建立该中转站的费用 每个 ...
- mysql更新密码为空
1.进入命令行 mysql -u root -p 'oldpassword'; 2 修改root用户的密码:mysql> update mysql.user set password=PASSW ...
- C#DataTable操作
] 在DataSet中添加DataTable DataSet.Tables.Add(DataTable) 实例: DataSet ds=new DataSet(); DataTable table=n ...
- dedecms flag标签属性
头条[h] flag='h' 推荐[c] flag='c' 幻灯[f] flag='f' 特荐[a] flag='a' 滚动[s] flag='s' 加粗[b] flag='b' 图片[p] flag ...
- php之分页类代码
/* 思路 1.把地址栏的URL获取 2.分析URL中的query部分--就是?后面传参数的部分 3.query部分分析成数组 4.把数组中的page单元,+1,-1,形成2个新的数组 5.再把新数组 ...
- discuz二次开发技巧
discuz二次开发技巧 二次开发大多时候知识设置和处理,如果能够获知模板文件获得的变量数组将大大提高我们的开发效率 获取页面已经定义的变量 <--{eval printf_r(get_defi ...