CSUFT 编译原理实验二LL(1)文法分析

  1 #include <cstdio>
  2 #include <cstring>
  3 #include <iostream>
  4 #include <stack>
  5 #include <queue>
  6 #include <map>
  7 #include <algorithm>
  8 #include <vector>
  9 
 10 using namespace std;
 11 
 12 char A[20];/*分析栈*/
 13 char B[20];/*剩余串*/
 14 char v1[20]= {'i','+','*','(',')','#'}; /*终结符  */
 15 char v2[20]= {'E','G','T','S','F'}; /*非终结符   */
 16 
 17 int j=0,b=0,top=0,l;/*L为输入串长度 */
 18    char x,ch;
 19     int k=1;
 20     int m,n;
 21     int flag=0,finish=0;
 22 
 23 typedef struct type/*产生式类型定义      */
 24 {
 25     char origin;/*大写字符  */
 26     char array[5];/*产生式右边字符 */
 27     int length;/*字符个数      */
 28 } type;
 29 
 30 type e,t,g,g1,s,s1,f,f1,cha;/*结构体变量  */
 31 type C[10][10];/*预测分析表      */
 32 
 33 void init()
 34 {
 35     e.origin='E';
 36     strcpy(e.array,"TG");
 37     t.origin='T';
 38     strcpy(t.array,"FS");
 39     g.origin='G';
 40     strcpy(g.array,"+TG");
 41     g1.origin='G';
 42     g1.array[0]='^';
 43     s.origin='S';
 44     strcpy(s.array,"*FS");
 45     s1.origin='S';
 46     s1.array[0]='^';
 47     f.origin='F';
 48     strcpy(f.array,"(E)");
 49     f1.origin='F';
 50     f1.array[0]='i';
 51 
 52     for(int m=0; m<=4; m++) /*初始化分析表*/
 53         for(int n=0; n<=5; n++)
 54             C[m][n].origin='N';/*全部赋为空*/
 55     /*填充分析表*/
 56     C[0][0]=e;
 57     C[0][3]=e;
 58     C[1][1]=g;
 59     C[1][4]=g1;
 60     C[1][5]=g1;
 61     C[2][0]=t;
 62     C[2][3]=t;
 63     C[3][1]=s1;
 64     C[3][2]=s;
 65     C[3][4]=C[3][5]=s1;
 66     C[4][0]=f1;
 67     C[4][3]=f;
 68 }
 69 
 70 void print()/*输出分析栈  */
 71 {
 72     int a;/*指针*/
 73     for(a=0; a<=top+1; a++)
 74         printf("%c",A[a]);
 75     printf("		");
 76 }
 77 
 78 void print1()/*输出剩余串*/
 79 {
 80     int j;
 81     for(j=0; j<b; j++) /*输出对齐符*/
 82         printf(" ");
 83     for(j=b; j<=l; j++)
 84         printf("%c",B[j]);
 85     printf("			");
 86 }
 87 int main()
 88 {
 89     init();
 90     //测试样例反例
 91     //i+i(i+i)#
 92     //i()#
 93     //正确样例
 94     //i+i*i#
 95 
 96 
 97     do/*读入分析串*/
 98     {
 99         scanf("%c",&ch);
100         if ((ch!='i') &&(ch!='+') &&(ch!='*')&&(ch!='(')&&(ch!=')')&&(ch!='#'))
101         {
102             printf("输入串中有非法字符
");
103             exit(1);
104         }
105         B[j]=ch;
106         j++;
107     }while(ch!='#');
108     l=j;/*分析串长度*/
109     ch=B[0];/*当前分析字符*/
110     A[top]='#';
111     A[++top]='E';/*'#','E'进栈*/
112     printf("步骤		分析栈 		剩余字符 		所用产生式 
");
113 
114         do{
115             flag = 0;
116         if(finish) break;
117         x=A[top--];/*x为当前栈顶字符*/
118         printf("%d",k++);
119         printf("		");
120 
121 
122 
123 
124         for( j=0; j<=5; j++) /*判断是否为终结符*/
125         {
126             if(x==v1[j])
127             {
128                 flag=1;
129                 break;
130             }
131         }
132             if(flag==1)/*如果是终结符*/
133             {
134                 if(x=='#')
135                 {
136                     finish=1;/*结束标记*/
137                     printf("acc!
");/*接受 */
138                     getchar();
139                     getchar();
140                     exit(1);
141                 }
142                 if(x==ch)
143                 {
144                     print();
145                     print1();
146                     printf("%c匹配
",ch);
147                     ch=B[++b];/*下一个输入字符*/
148                     flag=0;/*恢复标记*/
149                 }
150                 /*else //出错处理
151                 {
152                     print();
153                     print1();
154                     printf("%c出错
",ch);//输出出错终结符
155                     exit(1);
156                 }*/
157             }
158             else/*非终结符处理*/
159             {
160                 for(j=0; j<=4; j++)
161                     if(x==v2[j])
162                     {
163                         m=j;/*行号*/
164                         break;
165                     }
166                 for(j=0; j<=5; j++)
167                     if(ch==v1[j])
168                     {
169                         n=j;/*列号*/
170                         break;
171                     }
172                 cha=C[m][n];
173                 if(cha.origin!='N')/*判断是否为空*/
174                 {
175                     print();
176                     print1();
177                     printf("%c->",cha.origin);/*输出产生式*/
178                     int len = strlen(cha.array); //长度是array的长度
179                     for(j=0; j<len; j++)
180                         printf("%c",cha.array[j]);
181                     printf("
");
182                     for(j=(len-1); j>=0; j--) /*产生式逆序入栈*/
183                         A[++top]=cha.array[j];
184                     if(A[top]=='^')/*为空则不进栈*/
185                         top--;
186                 }
187                 else //出错处理 调整顺序
188                 {
189                     print();
190                     print1();
191                     printf("%c出错
",ch);//输出出错终结符
192                     exit(1);
193                 }
194 
195             }
196 
197         }while(top!=-1); //当分析栈不为空
198 
199     return 0;
200 }
View Code

送人玫瑰手留余香。。。。

原文地址:https://www.cnblogs.com/lmlyzxiao/p/5606594.html