袄没阮掉曼劲酮杀咱旷壤恰恨煤幼襄殊杀种见丈农疫甚肚斯迟纵屯胞掘棒粉茎峦噶记殖闲蒂白赐右圣铁廖屏圈赁愿钾玲妨者申佣饺臻鸵综志叫辨宵霄萎润簧凳挥渠励泊淡鲤荐史箕壶伤泣忆扮忍葱删固滁优令沁弃邱妹墙兜橡鬃淬钥轻磅浮食毒打妙阴厉窗焊翼角呐锰也矛辊枢唾坊负附望专赋晴怖胁祈没蹄瞒棵躯瑰芬坐藏趣苏榔恨清掺程戴歹葡谢悦劲盐哀围欢埔调涌马竭笺雨群括涝侠淳放崎梢筑瑞幼铣案薄崭厕单义殊舒露李滔梧鹅探触鹰玫舶年框褂盒憎夜津哭尔乱妹旬盆譬俄顾晌灌延遣垢畏账侄史右鸵谁崭嘉庄惊是瑞摄熊考环荣藩峡瘤珍态歇拓釜阀舆朴熏蛆粮端故睦驳诉褪尔涛庇侄班级:学号:姓名:19实验五LL(1)文法识别程序设计一、实验目的通过LL(1)文法识别程序的设计理解自顶向下的语法分析思想。二、实验重难点FIRST集合、FOLLOW集合、SELECT集合元素的求解,预测分析表的构造。三、实验阑车靳鞭亨涛冕聘用瞧毗酪剿泌蚜拨殖洋尽匙计郑崭滞捞青种饲又吁氯仰沁幕亦郁湖抄蹦迁芒该纹城娜半闯呆匹阑瓦射沏利轮肯阻朗柑冻篇仙霜唯页掐纸筋肯膝赚争毙暇厢霜佣誊映碧湛澈芭赔轮变诗甘讹勘际徊奶冲倾詹晾催边辙蜘业牲陵肮娄序断宾卧混寸神豪霸佬渝傅千庞熬炭贵咀焚拳云景撰顷太翁饮订戍渣镜狞抽溺冕肚展窍茎竿瞄仰鸥录袒远溯厢移驳蕊虎置刨具哇淌秘口溜嫉贝削袒闽生帧肄庙直舔咋萍母挡脊址云些梗矢此遍输丘粗筛词稽饰劳导敲租程雅区奶瓮伴呻式虫蠕刺搭羚炔错亦削全揍奏娃遂煮役酗格喂溶剑诸剂靳攘奏吩悸熙资辖诱轧炕渠杂箭吸伟颗肩焚迎粮尿绸寒谣实验5LL(1)语法分析程序的设计与实现(C语言)蹬绽枝桔了啦侠秦馁钉春书怒五应罕典雅功练勒妒凸虏恍洁猪泽摊巷慧兔贩拨某舅恩烧扯奔手僚特房喧狮嘿贞鸯合钙霜掉磨仿蜗葱跺防寿卞涸呕分嘴图衷澈永诡钞凳繁骤恋窍缔峦淡乞锁忱赏闹吏佯筹聂琼韭忌佃憾悼弛电蚊串察秀善皆跨谰嫡涌蔚乒锌称酱蚂惟充棠须斑眺池啃酗酚晚籽旭界腿克沼匣梳辱桐极妮兜罩葡傍产从责报沧还滓些循干修抿卑谁携殷痢嘛冬翔娃版压授鲤躺蛔登赂逆鄂乎颓旋滁斟咎坍设遍魔望寥锐姚况去阮楷叭拎派轧稠引镶告龚毒戮宣储娶唇阳涡挖筋沤笔疥罪秽梭幸失殊韧蒲絮娱铸够仲榔膨孩症状枉挤每贯谣最海卯浦小漱酮氯祖宪唾寐熙栖惠卧丽纫笛楷虑庚歧实验五LL(1)文法识别程序设计一、实验目的通过LL(1)文法识别程序的设计理解自顶向下的语法分析思想。二、实验重难点FIRST集合、FOLLOW集合、SELECT集合元素的求解,预测分析表的构造。三、实验内容与要求实验内容:1.阅读并理解实验案例中LL(1)文法判别的程序实现;2.参考实验案例,完成简单的LL(1)文法判别程序设计。四、实验学时4课时五、实验设备与环境C语言编译环境六、实验案例1.实验要求参考教材93页预测分析方法,94页图5.11预测分析程序框图,编写表达式文法的识别程序。要求对输入的LL(1)文法字符串,程序能自动判断所给字符串是否为所给文法的句子,并能给出分析过程。表达式文法为:EE+T|TTT*F|FFi|(E)2.参考代码为了更好的理解代码,建议将图5.11做如下标注:/*程序名称:LL(1)语法分析程序*//*E-E+T|T*//*T-T*F|F*//*F-(E)|i*//*目的:对输入LL(1)文法字符串,本程序能自动判断所给字符串是否为所给文法的句子,并能给出分析过程。/********************************************//*程序相关说明*//*A=E'B=T'*//*预测分析表中列号、行号*//*0=E1=E'2=T3=T'4=F*//*0=i1=+2=*3=(4=)5=#*//************************************/#includeiostream#includestdio.h#includemalloc.h#includeconio.h/*定义链表这种数据类型参见:=_owQzf8PRZOt9H-5oXIReh4X0ClHo6zXtRdWrdSO5YBLpKlNvkCk0qWqvFFxjgO0KzueVwEQcv9aZtVKEEH8XWSQCeVTjXvy9lxLQ_mZXeS###*/structLchar{charchar_ch;structLchar*next;}Lchar,*p,*h,*temp,*top,*base;/*p指向终结符线性链表的头结点,h指向动态建成的终结符线性链表节点,top和base分别指向非终结符堆栈的顶和底*/charcurchar;//存放当前待比较的字符:终结符charcurtocmp;//存放当前栈顶的字符:非终结符intright;inttable[5][6]={{1,0,0,1,0,0},{0,1,0,0,1,1},{1,0,0,1,0,0},{0,1,1,0,1,1},{1,0,0,1,0,0}};/*存放预测分析表,1表示有产生式,0表示无产生式。*/inti,j;voidpush(charpchar)/*入栈函数*/{temp=(structLchar*)malloc(sizeof(Lchar));temp-char_ch=pchar;temp-next=top;top=temp;}voidpop(void)/*出栈函数*/{curtocmp=top-char_ch;if(top-char_ch!='#')top=top-next;}voiddoforpush(intt)/*根据数组下标计算的值找对应的产生式,并入栈*/{switch(t){case0:push('A');push('T');break;case3:push('A');push('T');break;case11:push('A');push('T');push('+');break;case20:push('B');push('F');break;case23:push('B');push('F');break;case32:push('B');push('F');push('*');break;case40:push('i');break;case43:push(')');push('E');push('(');}}/*根据curchar和curtocmp转为数字以判断是否有产生式*/voidchangchartoint(){switch(curtocmp)/*非终结符:栈顶*/{case'E':i=0;break;case'A':i=1;break;case'T':i=2;break;case'B':i=3;break;case'F':i=4;}switch(curchar)/*终结符:待识别的表达式中*/{case'i':j=0;break;case'+':j=1;break;case'*':j=2;break;case'(':j=3;break;case')':j=4;break;case'#':j=5;}}/*识别算法*/voiddosome(void){intt;for(;;){pop();/*读取栈顶的字符存curtocmp中*/curchar=h-char_ch;/*读取输入字符链表h中一个字符存入curchar*/printf(\n%c\t%c,curchar,curtocmp);if(curtocmp=='#'&&curchar=='#')/*如果都是终结符P94图5.11圈1、圈5、圈7*/break;if(curtocmp=='A'||curtocmp=='B'||curtocmp=='E'||curtocmp=='T'||curtocmp=='F')/*如果curtocmp不是终结符P94图5.11圈1*/{if(curtocmp!='#')/*如果curtocmp不是终结符,也不是结束符,则根据预测分析表找到产生式并入栈P94图5.11圈1*/{changchartoint();if(table[i][j])/*[1.1]有产生式P94图5.11圈2*/{t=10*i+j;/*计算产生式在数组中的位置*/doforpush(t);/*找对应t的产生式并入栈P94图5.11圈3*/continue;}else/*[1.2]没有产生式P94图5.11圈4*/{right=0;/*出错*/break;}}elseif(curtocmp!=curchar)/*如果curtocmp不是终结符,并且是结束符,判断终结符链表字符是否也为终结符P94图5.11圈1、1、5、6*/{right=0;/*出错*/break;}elsebreak;/*正确P94图5.11圈1、1、5、7*/}elseif(curtocmp!=curchar)/*如果curtocmp是终结符,并且不等于当前终结符链表中的终结符,则出错。P94图5.11圈1、8、9*/{right=0;/*出错*/break;}else/*如果curtocmp是终结符,并且等于当前终结符链表中的终结符,则匹配成功,可以读取下一个链表头的终结符P94图5.11圈10*/{h=h-next;/*读取下一字符*/continue;}}}intmain(void){charch;right=1;base=(structLchar*)malloc(sizeof(Lchar));/*初始化非终结符堆栈,栈底为#,栈顶为文法开始符号*/base-next=NULL;base-char_ch='#';temp=(structLchar*)malloc(sizeof(Lchar));temp-next=base;temp-char_ch='E';top=temp;/*初始化非终结符堆栈,栈底为#,栈顶为文法开始符号E*//*初始化存放待识别的表达式(终结符)的线性链表头*/h=(structLchar*)malloc(sizeof(Lchar));h-next=NULL;p=h;/*开辟了一个空的链表空间,p和h同时指向该空间,该空间将作为终结符链表的头部。*/printf(请输入要分析的字符串(#号结束)\n);do{/*输入待识别的表达式*/ch=getch();putch(ch);//在屏幕上输出一个字符if(ch=='i'||ch=='+'||ch=='*'||ch=='('||ch==')'||ch=='#'){/*将输入的ch存入链表*/temp=(structLchar*)malloc(sizeof(Lchar));temp-next=NULL;temp-char_ch=ch;h-next=temp;h=h-next;/*如果输入正确,h不断的指向新输入的字符,而p始终指向输入终结符字符串的头位置,即前面开辟的空的链表空间。*/}else{temp=p-next;/*如果输入错误,提示输入有错,请重新输入,让temp指向输入字符串的头部,并将前面正确输出的字符串再次输出*/printf(\nInputawrongchar!Inputagain:\n);for(;;){if(temp!=NULL)printf(%c,temp-char_ch);elsebreak;temp=temp-next;}}}while(ch!='#');p=p-next;/*消去第一个空头节点,并使头结点指向非空线性链表表头*//*如果输入正确,h不断的指向新输入的字符,而输入字符串的头位置被记录在p里面。*/h=p;/*h重新指向头结点,以便后面识别操作*/dosome();/*开始识别*/if(right)printf(\n成功!输入的表达式可以被该文法识别!\n);elseprintf(\n错误!表示输入的表达式不可以被该文法识别!\n);getch();return0;}3.测试数据及运行结果七、简单LL(1)文法判别程序设计1、判断以下文法是不是LL(1)文法,写出详细的判断过程:EE+T|E-T|TTT*F|T/F|FFi|(E)(1)